本文的實(shí)現(xiàn)主要是基于 myclabs/php-enum 擴(kuò)展包。
今天來分享下如何管理 PHP 的枚舉類型。
一種常見的方式是,使用常量來代表枚舉類型
const YES = '是'; const NO = '否';
可以在這個(gè)基礎(chǔ)上更進(jìn)一步,將其封裝成類,以便于管理
class BoolEnum { const YES = '是'; const NO = '否'; }
現(xiàn)在,我們希望能通過方法來動(dòng)態(tài)調(diào)用對應(yīng)的枚舉類型
BoolEnum::YES(); // 是 BoolEnum::NO(); // 否
也可以批量獲取枚舉類型
BoolEnum::toArray(); // ['Yes' => '是', 'No' => '否']
下面來實(shí)現(xiàn)上面列舉的功能。
定義基本的枚舉基類,讓所有的枚舉類都繼承該抽象基類。
abstract class Enum { // 獲取所有枚舉類型 public static function toArray(){ // 通過反射獲取常量 $reflection = new ReflectionClass(static::class); $contants = $reflection->getConstants(); // 返回對應(yīng)的常量 return $contants; } // 動(dòng)態(tài)調(diào)用屬性 public static function __callStatic($name, $arguments) { $arr = static::toArray(); if(isset($arr[$name])){ return $arr[$name]; } throw new BadMethodCallException("找不到對應(yīng)的枚舉值 {$name}"); } } class BoolEnum extends Enum { const YES = '是'; const NO = '否'; }
利用反射,可以獲取到所有的枚舉類型。同時(shí),利用魔術(shù)方法則可以實(shí)現(xiàn)對屬性的動(dòng)態(tài)調(diào)用。這里要注意的是,反射會(huì)消耗較多的資源,因此,對 toArray 方法進(jìn)行重構(gòu),增加一個(gè)緩存變量來緩存獲取到的枚舉類型,避免重復(fù)使用反射。
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]; } }
現(xiàn)在考慮