问题

我知道PHP没有本机枚举.但我已经习惯于他们从Java世界.我想使用枚举作为一种方式来给予IDE的自动完成功能可以理解的预定义值.

常量会做到这一点,但是有命名空间冲突问题和(或实际上因为)它们是全局的.数组没有命名空间问题,但它们太模糊,它们可以在运行时被覆盖,IDE很少(从来没有)知道如何自动填充它们的键.

您通常使用哪些解决方案/解决方法?有没有人回忆起PHP的人有没有关于枚举的任何想法或决定?



解决方法

根据使用情况,我通常会使用 simple 类似以下内容:

abstract class DaysOfWeek
{
    const Sunday = 0;
    const Monday = 1;
    // etc.
}

$today = DaysOfWeek::Sunday;

但是,其他用例可能需要更多的常量和值的验证.根据以下关于反射的评论,以及其他一些说明,这里有一个扩展示例,可以更好地服务于更广泛的范围的案例:

abstract class BasicEnum {
    private static $constCacheArray = NULL;

    private static function getConstants() {
        if (self::$constCacheArray == NULL) {
            self::$constCacheArray = [];
        }
        $calledClass = get_called_class();
        if (!array_key_exists($calledClass, self::$constCacheArray)) {
            $reflect = new ReflectionClass($calledClass);
            self::$constCacheArray[$calledClass] = $reflect->getConstants();
        }
        return self::$constCacheArray[$calledClass];
    }

    public static function isValidName($name, $strict = false) {
        $constants = self::getConstants();

        if ($strict) {
            return array_key_exists($name, $constants);
        }

        $keys = array_map('strtolower', array_keys($constants));
        return in_array(strtolower($name), $keys);
    }

    public static function isValidValue($value, $strict = true) {
        $values = array_values(self::getConstants());
        return in_array($value, $values, $strict);
    }
}

通过创建一个扩展BasicEnum的简单枚举类,您现在可以使用方法进行简单的输入验证:

abstract class DaysOfWeek extends BasicEnum {
    const Sunday = 0;
    const Monday = 1;
    const Tuesday = 2;
    const Wednesday = 3;
    const Thursday = 4;
    const Friday = 5;
    const Saturday = 6;
}

DaysOfWeek::isValidName('Humpday');                  // false
DaysOfWeek::isValidName('Monday');                   // true
DaysOfWeek::isValidName('monday');                   // true
DaysOfWeek::isValidName('monday', $strict = true);   // false
DaysOfWeek::isValidName(0);                          // false

DaysOfWeek::isValidValue(0);                         // true
DaysOfWeek::isValidValue(5);                         // true
DaysOfWeek::isValidValue(7);                         // false
DaysOfWeek::isValidValue('Friday');                  // false

另一方面,任何时候,我在在数据不会改变的静态/ const类(例如枚举)中至少使用反射一次,我会缓存这些结果反射调用,因为每次使用新的反射对象将最终具有显着的性能影响(存储在多个枚举的关联数组中).

现在大多数人已经最终升级到至少5.3,并且 SplEnum 可用,这当然是一个可行的选择,只要你'记住在整个代码库中传统上不具有实际枚举实例的概念.在上面的例子中, BasicEnum DaysOfWeek 根本不能实例化,也不应该是.




相关问题推荐