静态方法调用绑定声明类而非调用类,self编译期绑定当前类,static运行期绑定初始调用类,parent仅限子类调用父类成员;new static()支持 late static binding,new self()则始终创建声明类实例。

静态方法调用时 :: 操作符绑定的是「声明类」而非「调用类」
PHP 中静态方法的解析发生在编译期,:: 左侧的类名(或 self、static、parent)决定了实际执行哪个方法体。它不看对象实例类型,也不看运行时 $this 是什么 —— 即使你用子类实例调用父类静态方法,只要写的是 ParentClass::method(),就一定执行父类里的定义。
常见错误现象:
– 子类重写了父类静态方法,但用 ParentClass::foo() 调用时,仍执行父类版本(这其实是预期行为);
– 误以为 $obj::foo() 会动态绑定到子类,结果发现它等价于 get_class($obj)::foo(),而该类若没重写,则继续向上查找;
– 在父类中写 self::bar(),本意想复用当前类逻辑,结果子类继承后调用时仍走父类的 self,不是子类的。
self、static、parent 的行为差异必须分清
三者都是作用域关键字,但绑定时机和目标完全不同:
-
self:编译期绑定,永远指向「当前代码所在类」(即写self::xxx那个类),不可被子类覆盖 -
static:运行期绑定(late static binding),指向「最初发起静态调用的那个类」,支持被子类重写覆盖 -
parent:只用于在子类中显式调用父类的成员,不能在父类里用parent::xxx(会报Cannot access parent:: when current class scope is not a class)
示例场景:父类定义了工厂方法,希望子类能用自己的静态方法创建实例:
立即学习“PHP免费学习笔记(深入)”;
class A {
public static function make() {
return new static(); // ← 这里用 static 才能返回 B 实例
}
}
class B extends A {}
var_dump(B::make()); // object(B)#1
如果把 new static() 换成 new self(),结果就是 object(A)#1 —— 这是高频翻车点。
子类调用同名静态方法时,:: 左侧写什么决定执行谁
假设:
class X { public static function say() { echo 'X'; } }
class Y extends X { public static function say() { echo 'Y'; } }
以下调用分别输出:
-
X::say()→X -
Y::say()→Y - 在
Y类内部写self::say()→Y - 在
Y类内部写parent::say()→X - 在
Y类内部写static::say()→Y(哪怕后续还有Z extends Y,Z::say()也会进Y的static::say())
注意:static:: 不是“当前类”,而是“静态调用发起者类”。如果在 Y 中定义了一个方法,里面写 static::say(),然后通过 Z::thatMethod() 调用,那 say() 就是 Z 的版本(如果 Z 有定义)。
不要依赖 $this::method() 做多态调度
虽然语法合法,但 $this::method() 实际等价于 static::method()(因为 $this 是对象,:: 会自动提取其所属类)。它看似“靠实例决定”,实则仍是 late static binding,且容易误导人认为它像普通方法一样走虚函数表。
真正需要运行时多态,请用普通(非静态)方法。静态方法天生不适合表达“根据对象类型动态选择行为”这个语义。
容易被忽略的一点:
– self 和 parent 在 trait 中的行为与在类中一致,但 static 在 trait 中指向的是使用该 trait 的类,不是 trait 自身 —— 这点在封装通用工厂逻辑时尤其关键。











