PHP 8.4 不支持 final const 语法,因类常量天生不可重写,final 修饰冗余且触发语法错误;可用 private const + 静态 getter 或 public readonly 属性替代,二者语义不同:const 编译期绑定,readonly 运行时单次赋值。

PHP 8.4 不支持 final const 语法——你不能用 final 修饰类常量。
这是个常见误解,源于把 final 类、final 方法和“只读常量”概念混在一起了。PHP 的类常量(const)天生不可重写,无论是否加 final,子类都无法覆盖父类的同名常量;所以语言层根本不需要、也不接受 final const 这种写法,它会直接报语法错误。
为什么 final const 是非法语法?
PHP 解析器在 8.4 中仍沿用原有语法规则:const 声明本身已是“最终绑定”,其值在编译期确定,且不允许被子类 override(注意:不是 redeclare,后者指重复定义,PHP 也不允许)。
因此,final 对 const 是冗余且无意义的修饰符,PHP 不提供该组合。
class ParentClass
{
public const VERSION = '1.0';
// public final const MODE = 'prod'; // ❌ Parse error: syntax error
}尝试这样写,会得到:
Parse error: syntax error, unexpected token "final", expecting identifier
PHP 8.4 中真正受 final 影响的常量相关行为
虽然不能写 final const,但有两个关键变化与“常量/不可变性”强相关:
-
只读属性(
readonly)继承限制强化:子类不得重声明父类的public readonly属性(否则 Fatal Error),这和常量的“不可覆盖”逻辑一致,但作用对象是属性而非常量; -
类常量仍可通过
self::/static::访问,但无法被修改或重新定义,这点从 PHP 5.0 就确立,8.4 未改动。
换句话说:你想保护一个值不被子类改写?用 const 就够了;想保护一个属性初始化后不再变?用 public readonly ——但别试图给 const 加 final。
立即学习“PHP免费学习笔记(深入)”;
替代方案:如何模拟“更严格”的常量控制?
如果你实际需要的是「运行时不可变 + 显式禁止子类干扰」,可考虑:
- 使用
private const+ 静态 getter 方法,隐藏实现细节; - 在构造或静态初始化中校验值合法性(例如结合
enum或match); - 利用 PHP 8.4 新增的
FFI或Dba\Connection等底层机制做只读内存映射(极少数场景); - 更现实的做法:靠文档 + 类型声明 + 静态分析工具(如 PHPStan)约束,而非语法糖。
class Config
{
private const DEFAULT_TIMEOUT = 30;
public static function getTimeout(): int
{
return self::DEFAULT_TIMEOUT;
}}
这样既避免子类接触常量名,又保留扩展灵活性。
真正容易被忽略的一点:很多人看到 readonly 属性和 const 都“不能改”,就默认它们语义等价。其实不然——const 是编译期绑定、无状态、无访问控制粒度;readonly 属性是运行时单次赋值、可带类型、可受访问修饰符影响、可参与对象生命周期。选哪个,取决于你要锁住的是「配置值」还是「实例状态」。











