【设计模式】桥接模式
1、概述
背景
我们在做一个大事件的时候,会把它分解成各个中事件,然后再把中事件分解成小事件。当我们完成各个小事件的时候,大事件自然就完成了。也就是是高层模块(大事件)应该依赖于底层模块(小事件)。
依赖倒置原则
依赖倒置原则指的是高层模块(稳定)不应该依赖于底层模块(变化),二者都应该依赖于抽象(稳定)。抽象(稳定)不应该依赖于实现(变化),实现应该依赖于抽象(稳定)。
举个例子,现在有一个需求,需要创建不同的图形,并且每个图形都有可能会有不同的颜色。我们可以利用继承的方式来设计类的关系如下图:
从图中我们可以很直观地看出箭头反转了,这就是依赖倒置原则的应用了。
但是,我们可以发现有很多的类,假如我们再增加一个形状或再增加一种颜色,就需要创建更多的类。
试想,在一个有多种可能会变化的维度的系统中,用继承方式会造成类爆炸,扩展起来不灵活。每次在一个维度上新增一个具体实现都要增加多个子类。为了更加灵活的设计系统,我们此时可以考虑使用桥接模式。
定义
将抽象与实现分离,使它们可以独立变化。它是用组合关系代替继承关系来实现,从而降低了抽象和实现这两个可变维度的耦合度。
结构
桥接(Bridge)模式包含以下主要角色:
- 抽象化(Abstraction)角色 :定义抽象类,并包含一个对实现化对象的引用。
- 扩展抽象化(Refined Abstraction)角色 :是抽象化角色的子类,实现父类中的业务方法,并通过组合关系调用实现化角色中的业务方法。
- 实现化(Implementor)角色 :定义实现化角色的接口,供扩展抽象化角色调用。
- 具体实现化(Concrete Implementor)角色 :给出实现化角色接口的具体实现。
2、实现思路
场景
需要开发一个电脑售卖平台,可以售卖不同名牌(华硕、戴尔、联想)的不同类型电脑(台式、笔记本)。该平台包含了两个维度,适合使用桥接模式。
/// <summary>
/// 名牌
/// </summary>
public interface IBrand
{
public void BrandDecision();
}
/// <summary>
/// 戴尔
/// </summary>
public class Dell : IBrand
{
public void BrandDecision()
{
Console.WriteLine("戴尔品牌");
}
}
/// <summary>
/// 联想
/// </summary>
public class Lenovo : IBrand
{
public void BrandDecision()
{
Console.WriteLine("联想品牌");
}
}
/// <summary>
/// 种类
/// </summary>
public interface IVariety
{
public void ReadMe();
}
/// <summary>
/// 台式机
/// </summary>
public class Desktop : IVariety
{
public void ReadMe()
{
Console.WriteLine("台式机");
}
}
/// <summary>
/// 笔记本
/// </summary>
public class Laptop : IVariety
{
public void ReadMe()
{
Console.WriteLine("笔记本");
}
}
public class Computer
{
private IBrand _brand = null;
private IVariety _variety = null;
public Computer(IBrand brand, IVariety variety)
{
this._brand = brand;
this._variety = variety;
}
public void StartUp()
{
this._brand.BrandDecision();
this._variety.ReadMe();
Console.WriteLine($"{this._brand.GetType().Name}的{this._variety.GetType().Name}开机");
}
}
class Program
{
static void Main(string[] args)
{
IBrand dell = new Dell();
IBrand asus = new ASUS();
IBrand lenovo = new Lenovo();
IVariety desktop = new Desktop();
IVariety laptop = new Laptop();
Computer computer = new Computer(dell,desktop);
computer.StartUp();
Computer computer1 = new Computer(asus, desktop);
computer1.StartUp();
Computer computer2 = new Computer(lenovo, laptop);
computer2.StartUp();
}
}
============================================
戴尔品牌
台式机
Dell的Desktop开机
华硕品牌
台式机
ASUS的Desktop开机
联想品牌
笔记本
Lenovo的Laptop开机
我们可以看出品牌都依赖于一个接口,种类也依赖于接口,这是依赖倒置的一种体现。我们将两个接口通过Computer类连接了起来,就像一座桥。如果现在还有一种新的品牌,我们只需要再定义一个类实现IBrand接口即可,其他类不需要发生变化,这也遵守了开闭原则。
3、优缺点
优点
- 桥接模式提高了系统的可扩充性,在两个变化维度中任意扩展一个维度,都不需要修改原有系统。
- 实现细节对客户透明
4 、使用场景
- 当一个类存在两个独立变化的维度,且这两个维度都需要进行扩展时。
- 当一个系统不希望使用继承或因为多层次继承导致系统类的个数急剧增加时。
- 当一个系统需要在构件的抽象化角色和具体化角色之间增加更多的灵活性时。避免在两个层次之间建立静态的继承联系,通过桥接模式可以使它们在抽象层建立一个关联关系。