PHP 枚举基类,使用类常量
作为枚举值,每一项均含有 key
、value
与 label
。
composer require lip/php-enum
一个枚举,通常包含:key
、value
,如 typescript
中的下列枚举:
enum Direction {
Up = 1,
Down = 2,
Left = 3,
Right = 4
}
其中 Up
、Down
、Left
、Right
为 key
,1
、2
、3
、4
为 value
。
一些常见的情形,还需要一个文字描述,特别是中文环境,将上面代码改造一下:
enum Direction {
Up = [1, '上'],
Down = [2, '下'],
Left = [3, '左'],
Right = [4, '右']
}
本库将这种文字描述,定义为枚举的 label
,即上面的上
、下
、左
、右
。
由于 PHP 本身不支持枚举,这里撰写一个类 lip\enum\Enum
,继承该类,定义类常量,作为枚举值。
每一个类常量均为一个数组
,其中,第一项为 value
,第二项为 label
,顺序不能变。
下例中,One、Two、Three 为 key
,1、2、3 是 value
,一、二、三是 label
。
<?php
use lip\enum\Enum;
/**
* 一个很有意思的枚举。
* @method static self One() One的函数说明
* @method static self Two() Two的函数说明
* @method static self Three() Three的函数说明
*/
final class Some extends Enum
{
private const One = [1, '一'];
private const Two = [2, '二'];
private const Three = [3, '三'];
}
定义一个使用 Some
枚举作为参数的函数,使用强类型
进行标注:
function useSome(Some $some) {
// ...
}
此时,如果用别的值,传递给上述函数 useSome
,则会报错,比如传递 Some::One
,由于它的值为 [1, '一']
,类型为 array
,而非 Some
,故而会报错。
Some
继承自 lip\enum\Enum
类,该类赋予 Some
一个很重要的能力:
将类常量定义,转换成对应的静态方法,调用此方法,会获得对应的
Some
类实例。
例如 Some::One()
对应的就是含有常量 Some::One
值的 Some
类实例,可以无障碍地传给 useSome
:
useSome(Some::One());
注意 Some::One()
是一个静态方法,它通过类 lip\enum\Enum
里的魔术方法 __callStatic 添加,有多少个类常量,就有多少个对应名字的静态方法。
那么 useSome
里如何获取枚举相应的 key
、value
、label
呢?调用相应的实例方法
即可,例如:
function useSome(Some $some)
{
$key = $some->key();
$value = $some->value();
$label => $some->label();
// $some 可以直接参与字符串连接运算,相当于 $some->value() 参与了运算,例如:
echo 'some value is: ' . $some;
// 相当于
echo 'some value is: ' . $some->value();
}
更多 lip\enum\Enum
类实例 & 静态方法参见下面的方法列表
部分。
强烈建议将类常量定义为
private
,以免被访问到,始终使用Some::ConstName()
的形式获取相应的枚举实例。
此用法不关注 label
,每个类常量均为单一值,此时 value
、label
相同,均为该常量值,举例来说:
/**
* 另一个很有意思的枚举。
* @method static self Haha() 哈哈大笑
* @method static self Bibi() You can you up, no can no bibi
*/
final class Other extends Enum
{
private const Haha = 'hh';
private const Bibi = 'bb';
}
// Other 实例的 value 与 label 是相同的,都是 hh
Other::Haha()->value() == Other::Haha()->label() // true
由于 lip\enum\Enum
类使用了魔术方法,导致在调用诸如 Some::One()
时,其结果不能正确地,被开发工具反射正确的类型,为使开发工具能正确地识别类型,可在枚举类的注释中,为每一个类常量增加一行 @method static self ConstName() 函数说明
,具体参见上面的例子。
基类 lip\enum\Enum
将构造函数设为 protected
,禁止通过构造函数实例化枚举类,请始终使用 Some::ConstName()
的形式。
方法名 | 说明 |
---|---|
key() | 获取枚举 key。 |
value() | 获取枚举 value。 |
label() | 获取枚举 label。 |
isXxx() | 动态方法,Xxx 为枚举 key。 |
__toString() | 无法被主动调用,在对实例本身进行字符串连接时,会被调用,此处实现为:(string)$this->value |
子类定义的类常量名,不要与👇的静态方法同名。
方法名 | 说明 |
---|---|
allConstants() | 获取全部常量列表,格式:[ key1 => [value1, label1], ... ] ,例如:[ 'One' => [1, '一'], 'Two' => [2, '二'], 'Three' => [3, '三'] ] 。 |
asList() | 作为列表返回,以便前端使用,返回值用类 js 格式表示:[ { key: key1, value: value1, label: label1 }, ... ] 。 |
parts() | 将实例的一部分,作为列表返回,返回值用类 js 格式表示:{ value, label } 。注意,这是实例方法。 |
allKeys() | 获取全部 key 列表。 |
allValues() | 获取全部 value 列表。 |
allLabels() | 获取全部 label 列表。 |
valueToLabel(mixed $value = null, $default = null) | 获取 $value 对应的 label,如不存在,返回 $default,若不传参数,返回整个 map ,例如:[1 => '一', 2 => '二', 3 => '三'] 。 |
valuesToLabels(array $values, $default = null) | 将一批 value 转换成对应的 label,若不存在,使用 $default 取代。 |
labelToValue(string $label = null, $default = null) | 获取 $label 对应的 value,如不存在,返回 $default,若不传参数,返回整个 map 。例如:['一' => 1, '二' => 2, '三' => 3] 。 |
labelsToValues(array $labels, $default = null) | 将一批 label 转换成对应的 value,若不存在,使用 $default 取代。 |
isValidKey(string $key) | 判断是否为合法的 key。 |
isValidValue(mixed $value) | 判断是否为合法的 value。 |
isValidLabel(string $label) | 判断是否为合法的 label。 |
fromKey(string $key) | 从 key 构建 Enum 实例,若不存在,则返回 null。 |
fromValue(mixed $value) | 从 value 构建 Enum 实例,若不存在,则返回 null。 |
isXxx(mixed $value) | 动态方法,Xxx 为枚举 key。 |
相同的类常量值,每次调用静态方法,均返回唯一的类实例,即:
Some::One() === Some::One() // true
Some::One() !== Some::One() // false
因此,判断一个枚举实例,是否为某个枚举时,可以这样:
if ($some === Some::One()) {
// ...
} else if ($some === Some::Two()) {
// ...
} else if ($some === Some::Three()) {
// ...
} else {
// ...
}
// 或者
switch ($some) {
case Some::One():
// ...
break;
case Some::Two():
// ...
break;
case Some::Three():
// ...
break;
default:
// ...
break;
}
同样,相等 == 也符合预期。
lip\enum\Enum
类实现了接口 JsonSerializable
,例如:
json_encode(Some::One()); // '{"key":"One","value":1,"label":"\u4e00"}'
运行 tests/runtest.sh
执行单元测试,已 100% 测试通过,请放心使用。
有任何建议或改进,请发 issue。
打 release Tag,转到 php-enum,就可以看到新版本了。