在PHP面向对象开发中,继承和组合都是实现代码复用的重要手段,但在实际面试和工程实践中,往往更推荐优先使用组合而非继承,这背后有着明确的设计逻辑和实际考量。

继承的常见问题与局限性
继承是PHP中很基础的特性,通过extends关键字可以让子类复用父类的属性和方法,看起来非常方便,但实际使用中会暴露不少问题。
首先是耦合性过高,子类会强依赖父类的实现,如果父类的方法逻辑发生变更,所有子类都可能受到影响。其次是灵活性不足,PHP是单继承语言,一个类只能继承一个父类,如果需要复用多个类的功能,继承就无法满足需求。另外继承容易违反里氏替换原则,子类如果重写父类方法改变了原有行为,可能会导致依赖父类的地方出现逻辑异常。
下面通过一个简单的继承示例来看问题:
<?php
// 父类:鸟类
class Bird {
public function fly() {
return "我会飞";
}
}
// 子类:麻雀,继承鸟类
class Sparrow extends Bird {
// 正常复用fly方法
}
// 子类:鸵鸟,继承鸟类
class Ostrich extends Bird {
// 鸵鸟不会飞,需要重写fly方法
public function fly() {
return "我不会飞";
}
}
$sparrow = new Sparrow();
echo $sparrow->fly(); // 输出:我会飞
$ostrich = new Ostrich();
echo $ostrich->fly(); // 输出:我不会飞
?>
上面的例子中,鸵鸟作为鸟类的子类却不会飞,重写fly方法后改变了父类的行为,这就违反了里氏替换原则,而且如果后续Bird类新增了其他方法,所有子类都会被强制关联,哪怕子类根本不需要这些方法。
组合的优势与实现方式
组合的核心思路是,一个类不直接继承另一个类,而是将需要复用的功能封装成独立的类,然后把该类的实例作为当前类的成员变量,通过调用成员对象的方法来实现功能复用。
组合的优势非常明显:第一,耦合性低,当前类只依赖成员对象暴露的接口,成员对象的内部实现变更不会影响当前类的使用;第二,灵活性高,一个类可以包含多个不同功能的成员对象,实现多维度功能复用;第三,符合单一职责原则,每个类只负责自己的核心功能,不会像继承那样让子类承担过多父类的职责。
我们用组合的方式改写上面的鸟类示例:
<?php
// 飞行能力接口
interface FlyAbility {
public function fly();
}
// 会飞的实现类
class CanFly implements FlyAbility {
public function fly() {
return "我会飞";
}
}
// 不会飞的的实现类
class CannotFly implements FlyAbility {
public function fly() {
return "我不会飞";
}
}
// 鸟类基础类
class Bird {
protected $flyAbility;
public function __construct(FlyAbility $flyAbility) {
$this->flyAbility = $flyAbility;
}
public function performFly() {
return $this->flyAbility->fly();
}
}
// 麻雀类,组合会飞的飞行能力
class Sparrow extends Bird {
public function __construct() {
parent::__construct(new CanFly());
}
}
// 鸵鸟类,组合不会飞的飞行能力
class Ostrich extends Bird {
public function __construct() {
parent::__construct(new CannotFly());
}
}
$sparrow = new Sparrow();
echo $sparrow->performFly(); // 输出:我会飞
$ostrich = new Ostrich();
echo $ostrich->performFly(); // 输出:我不会飞
?>
这个实现中,鸟类不需要关心飞行能力的具体实现,只需要持有对应的能力对象即可,后续如果需要新增其他飞行能力,只需要新增FlyAbility的实现类,不需要修改Bird类和它的子类,扩展性大大提升。
两者的适用场景区分
并不是说继承完全不能用,而是需要根据场景选择:
- 如果两个类之间是明确的is-a关系,比如麻雀是鸟,而且父类的所有方法子类都需要,且不会频繁变更,那么可以使用继承。
- 如果只是需要复用某个功能,类之间是has-a关系,比如鸟有飞行能力,或者需要复用多个类的功能,优先使用组合。
面试回答思路梳理
面试中被问到这类问题时,可以按照以下逻辑回答:首先说明组合和继承都是PHP中实现代码复用的方式,然后分别讲继承的局限性,比如耦合高、单继承限制、容易违反设计原则等,再讲组合的优势,比如低耦合、高灵活、易扩展,接着可以结合简单的代码示例说明两者的差异,最后补充两者的适用场景,说明不是完全否定继承,而是优先选择组合,这样回答会更完整,也能体现对设计原则的理解。