我希望能够做这样的事情:
class ThingIDs
{
const Something = 1;
const AnotherThing = 2;
}
$thing = 'Something';
$id = ThingIDs::$thing;
这行不通。有没有一种简单的方法来做一些等效的事情?请注意,我一直在上课;它在图书馆里,我不能重写。我正在编写在命令行中使用参数的代码,我真的很喜欢使用符号名称而不是id号。
使用constant()函数:
$id = constant("ThingIDs::$thing");
使用反射
$r = new ReflectionClass('ThingIDs');
$id = $r->getConstant($thing);
如果使用名称空间,则应将名称空间包含在类中。
echo constant('My'Application'ThingClass::ThingConstant');
Helper函数
你可以使用这样的功能:
function class_constant($class, $constant)
{
if ( ! is_string($class)) {
$class = get_class($class);
}
return constant($class . '::' . $constant);
}
它需要两个论点:
- 类名或对象实例
- 类常量名称
如果传递了对象实例,则会推断出其类名。如果使用PHP7,则可以使用::class
传递适当的类名,而不必考虑名称空间。
示例
class MyClass
{
const MY_CONSTANT = 'value';
}
class_constant('MyClass', 'MY_CONSTANT'); # 'value'
class_constant(MyClass::class, 'MY_CONSTANT'); # 'value' (PHP 7 only)
$myInstance = new MyClass;
class_constant($myInstance, 'MY_CONSTANT'); # 'value'
<?php
class Dude {
const TEST = 'howdy';
}
function symbol_to_value($symbol, $class){
$refl = new ReflectionClass($class);
$enum = $refl->getConstants();
return isset($enum[$symbol])?$enum[$symbol]:false;
}
// print 'howdy'
echo symbol_to_value('TEST', 'Dude');
如果您有对类本身的引用,那么您可以执行以下操作:
if (defined(get_class($course). '::COURSES_PER_INSTANCE')) {
// class constant is defined
}
我的问题与这个主题类似。当你有对象,但没有类名时,你可以使用:
$class_name = get_class($class_object);
$class_const = 'My_Constant';
$constant_value = constant($class_name.'::'.$class_const);
我知道我有点迟到了,但我希望这能有所帮助。
根据Phil的回答,我创建了一个可以扩展的默认枚举器类。
class DefaultEnum
{
static public function getConstantText(string $constant)
{
try {
// Get child class name that called this method
$child_class = get_called_class();
$reflection = new ReflectionClass($child_class);
$const = $reflection->getConstant($constant);
return $const;
} catch ('ReflectionException $e) {
// ...
}
}
}
class CustomEnum extends DefaultEnum
{
const something = 'abcd';
const something2 = 'ABCD';
}
你可以像这样调用这个方法
CustomEnum::getConstantText('something');
它将返回'abcd'
。
函数get_called_class()
是一个返回调用此方法的类名的函数,它专门适用于静态方法。
在这种情况下,$child_class
的值将是CustomEnum::class
。ReflectionClass接受字符串和对象作为参数。