定义
桥接模式是将抽象部分与它的实现部分分离,使他们都可以独立地变化。它是一种对象结构模式,又称为柄体模式或接口模式
理解桥接模式,重点需要理解如何将抽象化(Abstraction)与实现化(Implementation)脱耦,使得二者可以独立地变化。
抽象化:抽象化就是忽略一些信息,把不同的实体当作同样的实体对待。在面向对象中,将对象的共同性质抽取出来形成类的过程即为抽象化的过程。
实现化:针对抽象化给出的具体实现,就是实现化,抽象化与实现化是一对互逆的概念,实现化产生的对象比抽象化更具体,是对抽象化事物的进一步具体化的产物。
脱耦:脱耦就是将抽象化和实现化之间的耦合解脱开,或者说是将它们之间的强关联改换成弱关联,将两个角色之间的继承关系改为关联关系。桥接模式中的所谓脱耦,就是指在一个软件系统的抽象化和实现化之间使用关联关系(组合或者聚合关系)而不是继承关系,从而使两者可以相对独立地变化,这就是桥接模式的用意。
适用性
- 你不希望在抽象和他的实现部分之间有一个固定的邦定关系,如在程序的运行时刻实现部分应该可以被选择或者切换。
- 类的抽象以及他的视像都可以通过生成子类的方法加以扩充。这时bridge模式使你可以对不同的抽象接口和实现部分进行组合,并对他们进行扩充。
- 对一个抽象的实现部分的修改应该对客户不产生影响,即客户的代码不需要重新编译。
- 你想对客户完全隐藏抽象的实现部分。
- 你想在多个实现间 共享实现,但同时要求客户并不知道这一点。
结构
image.png
案例
现实生活中我们去电脑城买电脑,电脑城电脑的陈列结构如下图:
image.png
这种结构转换成代码如下:
//电脑类
class Computer{
}
//电脑类型类
Class Desktop extends Computer{
}
//电脑品牌类
class Apple extends Desktop{
}
以上代码存在多层继承关系,代码复杂度较高,且类之间的耦合度较高,不易于扩展。电脑、电脑类型、电脑品牌是三个独立的个体,不应该存在继承关系,应该通过组合连接在一起,桥接模式就是这类问题很好的解决方案
代码实现如下:
/**
* 品牌接口
* Interface Brand
*/
interface Brand
{
public function info();
}
/**
* 联想
* Class Lenovo
*/
class Lenovo implements Brand
{
public function info()
{
echo "联想";
}
}
/**
* 苹果
* Class Apple
*/
class Apple implements Brand
{
public function info()
{
echo "苹果";
}
}
/**
* 电脑抽象类
* Class Computer
*/
abstract class Computer
{
//电脑出厂自带品牌属性
//通过组合方式将品牌注入进来
/** @var Brand */
protected $brand;
public function __construct(Brand $brand)
{
$this->brand = $brand;
}
public function info()
{
$this->brand->info();
}
}
/**
* 台式电脑
* Class Desktop
*/
class Desktop extends Computer
{
public function info()
{
parent::info();
echo "台式机\n";
}
}
/**
* 笔记本电脑
* Class Laptop
*/
class Laptop extends Computer
{
public function info()
{
parent::info();
echo "笔记本\n";
}
}
//苹果笔记本
$apple = new Laptop(new Apple());
$apple->info();
//输出:苹果笔记本
//联想台式机
$computer = new Desktop(new Lenovo());
$computer->info();
//输出:联想台式机
其他应用场景
比如系统开发中的异常消息推送类,这个功能包括了消息渠道(QQ、微信、飞书、钉钉)、接收消息的用户(开发、运维...),使用桥接模式将用户与消息渠道组合实现异常消息推送功能