php类继承唯一语法是子类用extends关键字声明父类,仅支持单继承;private成员子类不可见,需用protected或getter访问;构造函数不自动继承,须显式调用parent::__construct()。

PHP类继承用 extends 关键字,不是 inherit 或 implements
PHP里实现继承只有一种语法:子类用 extends 关键字声明父类。别名、别写法、别加修饰符——错一个字符就报 Parse error: syntax error, unexpected token "extends" 或直接不生效。
常见错误是把接口实现和继承混用:implements 是对接口的契约承诺,extends 才是真正拿父类属性和方法过来用。如果父类是 abstract,子类必须实现所有抽象方法,否则会触发 Fatal error: Class contains abstract method and must therefore be declared abstract。
-
class Child extends Parent—— 正确,且只能单继承(PHP不支持多继承) -
class Child extends Parent1, Parent2—— 错误,PHP解析器直接拒掉 -
class Child implements Parent—— 语法虽不报错,但逻辑错:接口不能当类继承,Parent若非接口会报Class 'Parent' not found
父类的 private 成员在子类里不可见,别指望能直接调用
很多刚写继承的人卡在这儿:父类定义了 private $config,子类里写 $this->config 却是 null 或报 Notice: Undefined property。因为 private 是“仅本类可见”,子类不算“本类”。
想让子类能访问,得改用 protected —— 它允许子类和父类内部访问,又不暴露给外部调用。如果父类已经定死用 private,子类唯一能做的就是通过父类提供的 public 或 protected 方法间接操作,比如调用 $this->getConfig() 而不是直接读 $this->config。
立即学习“PHP免费学习笔记(深入)”;
- 父类
private function init()→ 子类无法parent::init()调用 - 父类
protected $data→ 子类可直接$this->data = [...] - 父类没提供
getter且字段是private→ 子类基本无解,只能重构父类或换设计
构造函数不会自动继承,子类要显式调用 parent::__construct()
PHP不会自动帮你跑父类的 __construct()。子类定义了自己的构造函数后,父类初始化逻辑就彻底失效——数据库连接没建、配置没加载、依赖没注入,全靠你手动补。
典型症状:子类对象创建后,父类里该初始化的属性全是 null 或默认值,但没报错,查半天才发现构造函数断层了。解决方式很简单:在子类 __construct() 开头加 parent::__construct(...),参数按父类要求传。
- 父类需要
__construct($host, $port),子类就得写parent::__construct($host, $port) - 如果子类构造函数参数更多,先处理自己的逻辑,再调父类,顺序别反
- 忘了调用?运行时通常没提示,只有业务出问题才暴露,难定位
重写父类方法时,return 类型和参数数量要对齐,否则可能被 IDE 误报或类型检查拦住
PHP本身对方法签名宽松,但启用 declare(strict_types=1) 或用 PHPStan/psalm 做静态分析时,子类重写方法若返回类型变窄(比如父类返回 array,子类写 string),或参数类型不兼容(父类接受 int,子类限定 positive-int),就会警告甚至报错。
更隐蔽的是参数数量:父类方法是 function save($id, $data),子类写成 function save($id, $data, $options = []) 合法,但反过来删参数(比如只留 $id)就违反LSP原则,运行时可能崩在父类调用处。
- 子类方法签名应至少兼容父类:参数类型不能更严格,返回类型不能更宽泛
- 加默认参数可以,删参数不行;加新参数可以,但得带默认值,避免调用方崩溃
- 用
@inheritDocPHPDoc 可帮 IDE 推导类型,但不解决实际兼容问题
extends 就万事大吉”,其实构造函数、可见性、方法签名这三块最容易漏掉细节。







