PHP中双冒号::是作用域解析操作符,用于访问类常量、静态属性和方法,以及实现后期静态绑定;必须用::而非->访问静态成员和常量,且self::编译时绑定、static::运行时绑定。

PHP 中的双冒号 :: 是作用域解析操作符(Scope Resolution Operator),不是“调用静态方法的语法糖”,而是用于明确指定在哪个类/作用域中访问成员——它既可用于静态成员,也可用于非静态常量、抽象方法占位、甚至后期静态绑定(static::)。
什么时候必须用 :: 而不能用 ->
当你要访问的是 const、static 属性或 static 方法时,对象实例无法使用 -> 访问这些成员。哪怕该静态方法内部用了 $this,也不能通过对象调用(PHP 会报 Fatal error: Using $this when not in object context)。
-
self::指向定义当前方法的那个类(编译时绑定) -
static::指向实际调用时的类(运行时绑定,支持后期静态绑定 LSB) -
parent::用于子类中调用父类被重写的静态成员 - 直接写
ClassName::是最明确的方式,不受继承上下文影响
:: 访问常量和静态属性的常见写法
类常量必须用 :: 访问,且不加 $;静态属性要加 $,但依然只能用 :: 读写(除非在类内部用 self::$prop)。
class Config {
const API_TIMEOUT = 30;
public static $retry_limit = 3;
}
echo Config::API_TIMEOUT; // ✅ 正确
echo Config::$retry_limit; // ✅ 正确
// echo Config->API_TIMEOUT; // ❌ 语法错误
// echo $config->retry_limit; // ❌ 静态属性不能用 ->
为什么 self:: 和 static:: 行为不同
这是最容易出 bug 的地方:如果你在父类里写 self::METHOD(),子类继承后调用,执行的仍是父类里的方法;而用 static::METHOD(),则会调用子类重写的版本。
立即学习“PHP免费学习笔记(深入)”;
class A {
public static function who() { return __CLASS__; }
public static function test() { return self::who(); } // 绑定到 A
public static function testLate() { return static::who(); } // 绑定到调用者
}
class B extends A {
public static function who() { return __CLASS__; }
}
echo A::test(); // A
echo A::testLate(); // A
echo B::test(); // A ← 容易误以为是 B
echo B::testLate(); // B ← 这才是你想要的多态效果
不能在普通函数或闭包里直接用 :: 调用类成员
:: 只能在类定义内部、或全局作用域中以 ClassName::xxx 形式使用。你不能在一个普通函数里写 self::xxx(因为没有类上下文),也不能在匿名函数里直接用 $this->method() 去触发静态调用逻辑——那会报错或行为异常。
- 在闭包中需要访问静态成员?显式传入类名字符串,再用
call_user_func([ClassName::class, 'method']) - 想动态调用?用
forward_static_call_array()或反射ReflectionMethod - 误把
new ClassName()::method()当成链式调用?PHP 不支持,会解析成(new ClassName())::method()并报错
真正要注意的不是“怎么写双冒号”,而是它背后的作用域绑定时机——self 看声明位置,static 看调用位置,ClassName 看你写死的是谁。多数静态工具类用 ClassName:: 最安全;涉及继承扩展时,static:: 才是让代码可覆盖的关键。











