PHP8类型系统怎么用_PHP8联合类型使用方式【操作】

2次阅读

php8联合类型需遵循语法约束:void不可参与联合,mixed不能与其他类型联合,可空类型必须显式写为T|NULL;参数和返回值均支持联合类型,但返回值校验更严格且不接受隐式转换;php8.1起才支持属性联合类型,且需显式声明|null才能赋null默认值。

PHP8类型系统怎么用_PHP8联合类型使用方式【操作】

PHP8联合类型怎么写才不报错

PHP8的联合类型(union Types)不是“多个类型随便写”,必须满足语法和语义约束,否则直接Parse Error。最常见错误是把voidmixednull混在联合中没处理好。

  • void不能出现在联合类型里——它本身不是类型,只是返回值声明,写function foo(): void|int会报Parse error: syntax error
  • mixed可以单独用,但不能和其他类型联合(如mixed|String非法),PHP认为它已涵盖全部可能
  • 如果想表示“字符串或空值”,必须显式写string|null,不能依赖?string自动展开——?string是可空类型语法糖,等价于string|null,但仅限单个类型前加?
  • 联合类型中类型顺序无关,但PHP会按声明顺序做运行时校验,不影响性能,只影响错误提示里的类型排列

函数参数和返回值联合类型的写法差异

参数和返回值都支持联合类型,但实际行为有微妙区别:参数类型检查在调用时触发,返回值检查在函数退出时触发;且返回值校验更严格——哪怕返回null而声明里没写|null,就会Fatal error。

  • 参数联合类型允许隐式转换失败时抛TypeError,比如function bar(int|Float $x)"123"会报错,不会自动转为int
  • 返回值联合类型不接受隐式转换,哪怕return "42"面对int|float也会挂掉,必须显式(int)"42"或改声明
  • 类方法继承时,子类重写方法的返回联合类型不能比父类更宽(LSP限制),例如父类声明string|int,子类只能缩小为string或保持一致,不能扩成string|int|bool

联合类型和PHP7的@var注解冲突吗

不冲突,但ide和静态分析工具(如PHPStan、Psalm)对两者的理解不同:PHP8原生联合类型是运行时强制的,而@var只是注释,不会影响执行。混用时容易让工具误判。

  • 如果已用PHP8联合类型,就别再写@var int|string $x——PHPStan会警告“redundant PHPDoc”
  • @param注解仍可补充说明,比如@param int|float $rate 百分比数值,支持小数,这是合法且有用的
  • 注意phpdoc中的联合类型写法是int|float,和PHP8语法一致;但旧版工具可能不识别Staticself参与的联合,比如self|OtherClass在PHPStan 1.5+才稳定支持

联合类型在属性上怎么用才安全

PHP8.1起才支持属性联合类型(PHP8.0只支持参数和返回值),所以用PHP8.0写public int|string $prop;会直接报错。升级前务必确认版本。

立即学习PHP免费学习笔记(深入)”;

  • 属性联合类型不允许默认值为null,除非显式包含|null,例如public string|null $name = null;合法,但public string|int $id = null;会报TypeError
  • 构造函数参数类型和属性类型不自动对齐,即使写了public function __construct(public int|string $id) {},也不能省略属性声明里的类型(PHP8.1+支持属性提升,但类型仍需显式写出)
  • 序列化/反序列化不受联合类型保护——unserialize()出来的对象属性可能违反类型声明,运行时不会立刻报错,直到下次访问该属性才触发类型检查

联合类型看着简单,但int|floatfloat|int虽然等价,一旦涉及泛型(如ArrayObject<int></int>)或反射(ReflectionParameter::getType()返回对象),顺序会影响__toString()输出和工具链解析,这点很容易被忽略。

text=ZqhQzanResources