PHP如何实现类的继承_PHP实现类继承的语法【语法】

1次阅读

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

PHP如何实现类的继承_PHP实现类继承的语法【语法】

PHP类继承用 extends 关键字,不是 inheritimplements

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,子类唯一能做的就是通过父类提供的 publicprotected 方法间接操作,比如调用 $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原则,运行时可能崩在父类调用处。

  • 子类方法签名应至少兼容父类:参数类型不能更严格,返回类型不能更宽泛
  • 加默认参数可以,删参数不行;加新参数可以,但得带默认值,避免调用方崩溃
  • @inheritDoc PHPDoc 可帮 IDE 推导类型,但不解决实际兼容问题

子类不是父类的“副本”,而是扩展关系;很多问题出在假设“写了 extends 就万事大吉”,其实构造函数、可见性、方法签名这三块最容易漏掉细节。

text=ZqhQzanResources