通八洲科技

php中self::能调用子类重写的方法吗_静态绑定与重写关系【介绍】

日期:2026-01-01 00:00 / 作者:看不見的法師
self:: 不调用子类重写的方法,因其在编译期绑定到当前代码所在类,而非运行时对象类;static:: 才支持后期静态绑定,可调用子类重写的方法。

self:: 不能调用子类重写的方法 —— 它始终绑定到**当前书写该代码的类**,而非运行时实际对象的类。这是 PHP 静态绑定(early binding)的典型表现,和 static:: 的后期静态绑定(late static binding)有本质区别。

为什么 self:: 不会调用子类重写的方法?

因为 self:: 在编译期就确定了目标类,不随继承链动态变化。哪怕你在父类里写 self::foo(),子类继承并重写了 foo(),只要调用点在父类定义中,self:: 仍指向父类本身。

self::static:: 在方法调用上的关键差异

二者都可用于静态方法调用,但绑定时机不同:

class Parent {
    public static function who() {
        echo __CLASS__;
    }
    public static function testSelf() {
        self::who();   // 输出 Parent
    }
    public static function testStatic() {
        static::who(); // 输出 Child(若 Child::testStatic() 被调用)
    }
}

class Child extends Parent {
    public static function who() {
        echo __CLASS__;
    }
}

Child::testSelf();   // 输出:Parent
Child::testStatic(); // 输出:Child

哪些情况会让 self:: 看似“调用了子类方法”?

那通常不是 self:: 的功劳,而是以下几种常见误判:

什么时候该坚持用 self::

当你**明确需要锁定到当前类的行为**,防止被继承破坏逻辑时:

真正容易被忽略的点是:很多开发者以为把 self:: 换成 static:: 就能“支持继承”,却没检查目标方法是否为 public/protected、是否被正确重写、以及是否在静态上下文中被调用 —— 后者一旦出错,直接抛 Strict StandardsFatal error: Cannot make static method non-static