PHP 列挙型の管理と設計

藏色散人
藏色散人転載
2020-02-13 13:42:543555ブラウズ

この記事の実装は主に 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);
    }
}

これにより、ユーザーが不正な列挙型を使用することを回避できます。 value

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

またはパラメータ型修飾として

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

PHP は型付けが弱い言語であるため、パラメータ修飾が不十分な場合は多くの予期しないエラーが発生します。列挙型クラスを使用することで、パラメータ制限の機能がさらに強化されます。強化されると同時に、列挙型の管理がより便利になり、統合されます。

以上がPHP 列挙型の管理と設計の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事はlearnku.comで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。