首页后端开发PHPPHP 枚举类型的管理与设计

PHP 枚举类型的管理与设计

时间2024-02-02 01:44:03发布访客分类PHP浏览514
导读:收集整理的这篇文章主要介绍了PHP 枚举类型的管理与设计,觉得挺不错的,现在分享给大家,也给大家做个参考。本文的实现主要是基于 myclabs/php-enum 扩展包。今天来分享下如何管理 PHP 的枚举类型。一种常见的方式是,使用常量来...
收集整理的这篇文章主要介绍了PHP 枚举类型的管理与设计,觉得挺不错的,现在分享给大家,也给大家做个参考。本文的实现主要是基于 myclabs/php-enum 扩展包。

今天来分享下如何管理 PHP 的枚举类型。

一种常见的方式是,使用常量来代表枚举类型

const YES = '是';
    const NO = '否';
    

可以在这个基础上更进一步,将其封装成类,以便于管理

class BoolEnum {
        const YES = '是';
        const NO = '否';
}
    

现在,我们希望能通过方法来动态调用对应的枚举类型

BoolEnum::YES();
     // 是BoolEnum::NO();
     // 否

也可以批量获取枚举类型

BoolEnum::toArray();
     // ['Yes' =>
     '是', 'No' =>
     '否']

下面来实现上面列举的功能。

定义基本的枚举基类,让所有的枚举类都继承该抽象基类。

abstract class Enum{
       // 获取所有枚举类型    public static function toArray(){
            // 通过反射获取常量        $reflection = new \ReflectionClass(static::class);
            $contants = $reflection->
    getConstants();
            // 返回对应的常量        return $contants;
    }
    //  动态调用属性    public static function __callStatic($name, $arguments)    {
            $arr = static::toArray();
        if(isset($arr[$name])){
                return $arr[$name];
        }
        throw new \BadMethodCallException("找不到对应的枚举值 {
$name}
    ");
    }
}
class BoolEnum extends Enum{
        const YES = '是';
        const NO = '否';
}
    

利用反射,可以获取到所有的枚举类型。同时,利用魔术方法则可以实现对属性的动态调用。这里要注意的是,反射会消耗较多的资源,因此,对 toArray 方法进行重构,增加一个缓存变量来缓存获取到的枚举类型,避免重复使用反射。

abstract class Enum{
           PRotected static $cache = [];
    public static function toArray(){
            $class = static::class;
        // 第一次获取,就通过反射来获取        if(! isset(static::$cache[$class])){
                $reflection = new \ReflectionClass(static::class);
                static::$cache[$class] = $reflection->
    getConstants();
        }
            return static::$cache[$class];
    }
}
    

现在考虑更多的使用场景,比如用实例来代表特定枚举类型

$yes = new BoolEnum("是");
    echo $yes;
     // "是"

实现如下

abstract Enum {
        protected $value;
    public function __construct($value)    {
           if ($value instanceof static) {
                $value = $value->
    getValue();
        }
            if(! $this->
isValid($value)){
                throw new \unexpectedvalueexception("$value 不属于该枚举值" . static::class);
        }
            $this->
    value = $value;
    }
    // 获取实例对应的键    public function getKey(){
            return array_seArch($this->
    value, static::toArray(), true);
    }
    // 获取实例对应的值    public function getValue()    {
            return $this->
    value;
    }
    // 允许字符串形式输出    public function __toString()    {
            return $this->
    value;
    }
    // 验证值是否合法    public function isValid($value)    {
          $arr = static::toArray();
          return in_array($value, $arr, true);
    }
    // 验证键是否合法    public function isValidKey($key)    {
          $arr = static::toArray();
          return array_key_exists($key, $arr);
    }
}
    

这样做可避免用户使用非法的枚举类型的值

$user->
    banned = '非法值';
      // 可能不会报错$yes = new BoolEnum("非法值");
     // 将会抛出异常$user->
    banned = $yes;
    

或者作为参数类型限定

function setUserStatus(BoolEnum $boolEnum){
        $user->
    banned = $boolEnum;
}
    

PHP 作为一门弱类型语言,参数限定的不足会导致很多不可预期的错误发生,通过使用枚举类,我们进一步加强了参数限定的功能,同时,管理枚举类型也更加的方便统一。

以上就是PHP 枚举类型的管理与设计的详细内容,更多请关注其它相关文章!

声明:本文内容由网友自发贡献,本站不承担相应法律责任。对本内容有异议或投诉,请联系2913721942#qq.com核实处理,我们将尽快回复您,谢谢合作!


若转载请注明出处: PHP 枚举类型的管理与设计
本文地址: https://pptw.com/jishu/596196.html
PHP函数库之类与对象详解 怎么循环查看php-fpm的内存占用情况

游客 回复需填写必要信息