类常量必须用::访问,普通常量(define或全局const定义)不可用::,否则报错;::左侧须为类/接口/trait名称或可变类名,支持self、static、parent及命名空间限定名。

类常量必须用 :: 访问,普通常量不能用
PHP 中只有 class 定义的常量(即类常量)才支持用作用域操作符 :: 访问;而通过 define() 或 const 在全局作用域定义的常量,**不能**用 ::,否则会报 Fatal error: Uncaught Error: Undefined class constant。
常见错误写法:
define('API_TIMEOUT', 30);
echo API_TIMEOUT::class; // ❌ 错误:不是类,没有 :: 操作符
echo API_TIMEOUT::API_TIMEOUT; // ❌ 同样报错
正确做法是直接使用常量名:
echo API_TIMEOUT; // ✅ 输出 30
:: 只能用于类、接口、trait 的常量和静态成员
作用域操作符 :: 的左侧必须是一个有效的「命名空间限定名称」或「类/接口/trait 名称」,不能是变量、表达式或字符串(除非用 ClassName::CONST_NAME 这种字面量形式)。
立即学习“PHP免费学习笔记(深入)”;
-
MyClass::MY_CONST✅ 正确,类名字面量 + 常量名 -
$class = 'MyClass'; $class::MY_CONST✅ PHP 5.3+ 支持“可变类名”语法 -
self::MY_CONST✅ 在类内部访问自身常量 -
static::MY_CONST✅ 支持后期静态绑定(LSP),子类重写时指向实际调用类 -
parent::MY_CONST✅ 访问父类定义的常量(即使子类未重写)
注意:self 和 static 行为不同 —— self 是编译时绑定,static 是运行时绑定。若子类覆盖了同名常量,self:: 仍返回父类值,static:: 返回子类值。
类常量定义位置与可见性限制
类常量从 PHP 5.6 开始支持表达式(如 1 ),但依然不支持函数调用或变量;PHP 7.1+ 引入 private/protected 类常量(此前所有类常量默认 public)。
定义示例:
class Config {
const MODE = 'prod';
public const VERSION = '2.1.0';
protected const DEBUG = false;
private const SECRET_SALT = 'abc123';
public function show() {
echo self::MODE; // ✅ OK
echo static::VERSION; // ✅ OK(public 可被继承访问)
echo self::DEBUG; // ✅ OK(protected 允许类内访问)
// echo self::SECRET_SALT; // ❌ Fatal error:private 常量不可在类外访问,包括本类方法中?等等——实际上可以!✅ 正确:private 类常量可在定义它的类内部任意位置访问,包括方法中。
}
}
外部访问限制:
-
Config::MODE✅ public,允许 -
Config::DEBUG❌ protected,报Cannot access protected const -
Config::SECRET_SALT❌ private,报Cannot access private const
命名空间下类常量的完整引用路径
带命名空间的类,必须用完全限定名或 use 导入后使用别名,否则 :: 无法解析。
错误写法(未导入且未写全路径):
use My\Package\Foo; // ... echo Foo::BAR; // ✅ OK(已 use) echo \My\Package\Foo::BAR; // ✅ OK(全限定名) echo Package\Foo::BAR; // ❌ 缺少根命名空间 \,会被当成相对命名空间
特别注意:如果当前文件有命名空间声明(如 namespace App;),而你写了 Foo::BAR 且没 use,PHP 会尝试在 App\Foo 下找类,而不是全局 Foo。
建议统一用全限定名或显式 use,避免歧义。
类常量本身不参与命名空间解析 —— 它们属于类,而类名才是命名空间作用的对象。











