必须用::访问类常量和静态属性,因PHP语法硬性禁止用->操作符访问类层级成员;::用于类作用域,->仅限对象实例上下文。

为什么必须用 :: 访问类常量和静态属性
因为 PHP 不允许用对象实例(->)访问类常量或静态属性——这不是风格选择,而是语法硬性限制。试图写 $obj::CONST_NAME 或 $obj->STATIC_PROP 会直接报错:Fatal error: Uncaught Error: Access to undeclared static property 或类似提示。
根本原因在于:类常量和静态属性属于「类本身」,而非某个具体对象。它们在类加载时就已存在,不依赖实例化。而 -> 操作符只作用于对象上下文,天然无法触达类层级的符号空间。
:: 和 self::、static:: 的区别在哪
三者都用双冒号,但绑定时机和语义完全不同:
-
self::在定义时就绑定到当前类,不会随继承改变 —— 适合明确要锁定本类逻辑的场景 -
static::支持后期静态绑定(LSB),运行时指向实际调用的类 —— 更灵活,尤其在继承链中重载常量/静态属性时必须用它 - 直接写
ClassName::是最明确的硬编码引用,无动态性,但可读性强、IDE 友好
常见错误是把 self:: 当成 static:: 用。比如父类里写 self::VERSION,子类覆盖了 VERSION 常量,结果还是返回父类的值 —— 这不是 bug,是 self 的设计行为。
访问静态属性时漏掉 $ 会怎样
这是高频低级错误:写成 MyClass::MY_STATIC 而不是 MyClass::$MY_STATIC。PHP 会把它当成常量查找,找不到就报 Notice: Use of undefined constant MY_STATIC,甚至可能意外匹配到全局常量(如果恰好有同名 define())。
规则很简单:
- 类常量不用
$:MyClass::DEFAULT_LIMIT - 静态属性必须带
$:MyClass::$counter - 静态方法不用
$:MyClass::reset()
混淆的本质是 PHP 把常量和变量放在不同符号表里,:: 后面跟什么,解析器就查什么表。
性能与兼容性影响几乎为零,但写法决定可维护性
用 :: 访问静态成员本身没有运行时开销,PHP 在编译阶段就完成了符号解析。真正影响开发体验的是命名和引用方式:
- 避免在类内部大量硬编码类名(如
Config::DB_HOST),改用self::或static::提高复用性 - PHP 8.2+ 开始对未声明的静态属性触发
Deprecated警告,所以public static $x;必须显式声明,不能靠动态赋值“创建” - 类常量从 PHP 7.1 起支持表达式(如
const MAX = self::MIN * 2;),但依然不能用变量或函数调用初始化
最易被忽略的一点:静态属性是跨请求共享的(在 CLI 或长生命周期 SAPI 如 Swoole 中),但 Web FPM 场景下每个请求是独立进程,所以别误以为“静态属性能存用户数据”——它只在单次执行内有效。











