本文的实现主要是基于 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 作为一门弱类型语言,参数限定的不足会导致很多不可预期的错误发生,通过使用枚举类,我们进一步加强了参数限定的功能,同时,管理枚举类型也更加的方便统一。

更多相关文章

  1. 了解PHP中的8个魔术常量
  2. 通过实例解析PHP数据类型转换方法
  3. php魔术常量教程
  4. php中定义常量的两种方法
  5. php魔术常量使用方法
  6. php八大数据类型有哪些

随机推荐

  1. 如果让我重新设计一款Android App
  2. Android Studio中使用com.android.suppor
  3. Android应用程序启动Binder线程源码分析
  4. Android WebView 用法
  5. 【Android开机启动Activity或者Service方
  6. 定制自己的 Android(安卓)Dialog 信息提
  7. Android常用开源库(模块部分)
  8. 10.Android之测试代码实现步骤
  9. Android客户端通过socket与服务器通信
  10. 离线下载android sdk