子类重写父类方法未被调用最常见原因是方法签名不一致、修饰符缺失、拼写错误、误用self::、父类方法为final或父类构造函数未显式调用;静态方法中应使用static::而非self::以支持晚期静态绑定。

子类重写了父类方法但没被调用
最常见的情况是:你以为子类的 save() 被执行了,实际运行的还是父类的 save()。检查是否漏掉了 public、protected 修饰符,或子类方法名拼写错误(比如 svae())。PHP 不会报错,只是静默跳过——这和 Python 或 Java 不同,它不会强制要求重写签名一致。
- 确认子类方法声明与父类完全一致:
public function save($data)不能写成public function save($data = [])(除非父类也带默认值) - 检查是否在子类中误用了
self::save()—— 这会绑定到当前类字面量,应改用$this->save() - 如果父类方法是
final,子类无法覆盖,会直接报Fatal error: Cannot override final method
父类构造函数没被显式调用
PHP 不自动调用父类 __construct()。如果你在子类写了构造函数但没手动调 parent::__construct(),父类初始化逻辑就彻底跳过了,常导致依赖属性为空、数据库连接未建立等问题。
class User extends Model
{
public function __construct($id)
{
// ❌ 错误:只初始化了子类逻辑,父类构造函数被跳过
$this->id = $id;
}
}
class User extends Model
{
public function __construct($id)
{
// ✅ 正确:先让父类完成初始化
parent::__construct();
$this->id = $id;
}
}
使用 get_class_methods() 和 debug_backtrace() 快速定位调用链
当不确定哪个类的方法真正被触发时,别靠猜。在疑似入口处插入调试代码,比翻十几层继承更可靠。
-
get_class_methods($this)返回当前实例所有可调用方法(含继承来的),能立刻看出子类是否成功覆盖 - 在目标方法开头加
error_log(print_r(debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS), true));,查看调用栈里是谁在驱动这次执行 - 注意:如果用了 trait,
get_class_methods()也会包含 trait 中的方法,但debug_backtrace()显示的是实际调用位置
静态方法继承陷阱:self vs static
静态方法中用 self:: 会锁定到定义该方法的类,而不是调用者类。这是 PHP “晚期静态绑定”最易出错的地方。
立即学习“PHP免费学习笔记(深入)”;
class A
{
public static function who() { return __CLASS__; }
public static function test() { return self::who(); } // ❌ 绑定到 A
}
class B extends A
{
public static function who() { return __CLASS__; }
}
echo B::test(); // 输出 "A",不是 "B"
改成 static::who() 就能正确输出 B。只要涉及静态方法继承,无条件检查是否用了 static:: 替代 self::。
继承链越深,这个点越容易被忽略——尤其是团队协作时,有人改了父类静态方法,却没同步更新子类里的 self 引用。











