加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 站长学院 > PHP教程 > 正文

PHP 枚举类型的管理与设计

发布时间:2020-12-13 21:04:01 所属栏目:PHP教程 来源:网络整理
导读:本文的实现主要是基于?myclabs/php-enum?扩展包。 今天来分享下如何管理 PHP 的枚举类型。 一种常见的方式是,使用常量来代表枚举类型 const YES = '是';const NO = '否'; 可以在这个基础上更进一步,将其封装成类,以便于管理 class BoolEnum { const YES =
本文的实现主要是基于?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 作为一门弱类型语言,参数限定的不足会导致很多不可预期的错误发生,通过使用枚举类,我们进一步加强了参数限定的功能,同时,管理枚举类型也更加的方便统一。

更多学习内容请访问:

腾讯T3-T4标准精品PHP架构师教程目录大全,只要你看完保证薪资上升一个台阶(持续更新)

?

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读