当前位置:首页 > > 嵌入式云IOT技术圈
[导读]桥接模式(Bridge Pattern)是将抽象部分与它的实现部分分离,使它们都可以独立地变化。

桥接模式(Bridge Pattern)是将抽象部分与它的实现部分分离,使它们都可以独立地变化。

1

模式结构

UML 结构图:




  • Abstraction(抽象类):用于定义抽象类的接口,并且维护一个指向 Implementor 实现类的指针。它与 Implementor 之间具有关联关系。

  • RefinedAbstraction(扩充抽象类):扩充由 Abstraction 定义的接口,在 RefinedAbstraction 中可以调用在 Implementor 中定义的业务方法。

  • Implementor(实现类接口):定义实现类的接口,这个接口不一定要与 Abstraction 的接口完全一致,事实上这两个接口可以完全不同。

  • ConcreteImplementor(具体实现类):实现了 Implementor 定义的接口,在不同的 ConcreteImplementor 中提供基本操作的不同实现。在程序运行时,ConcreteImplementor 对象将替换其父类对象,提供给 Abstraction 具体的业务操作方法。

2

优缺点


优点:

  • 分离抽象和实现部分。桥接模式使用“对象间的关联关系”解耦了抽象和实现之间固有的绑定关系,使得抽象和实现可以沿着各自的维度来变化。即抽象和实现不再在同一个继承层次结构中,而是“子类化”它们,使它们各自都具有自己的子类,以便可以进行任意组合,从而获得多维度的组合对象。

  • 在很多情况下,桥接模式可以取代多层继承方案。多层继承违背了“单一职责原则”,复用性较差,且类的个数非常多。所以相比起来,桥接模式更好,它极大地减少了子类的个数。

  • 提高了系统的可扩展性,在两个变化维度中任意扩展一个维度,都不需要修改原有系统,符合“开闭原则”。

缺点:

  • 增加了系统的理解与设计难度,由于关联关系建立在抽象层,要求开发者一开始就针对抽象层进行设计与编程。

  • 需要能正确识别出系统中两个独立变化的维度,因此使用范围具有一定的局限性,如何正确识别两个独立维度也需要一定的经验积累。

3

适用场景

  • 如果一个系统需要在抽象化和具体化之间增加更多的灵活性,避免在两个层次之间建立静态的继承关系,通过桥接模式可以使它们在抽象层建立一个关联关系。

  • “抽象部分”和“实现部分”可以以继承的方式独立扩展而互不影响,在程序运行时,可以动态地将一个抽象化子类的对象和一个实现化子类的对象进行组合,即系统需要对抽象化角色和实现化角色进行动态耦合。

  • 一个系统存在多个(≥ 2)独立变化的维度,且这多个维度都需要独立进行扩展。

  • 对于那些不希望使用继承或因为多层继承导致系统类的个数急剧增加的系统,桥接模式尤为适用。

4

案例分析

开关和电器



电器是生活中必不可少的东西,几乎每家每户都有,例如:电视、风扇、电灯 ...... 虽然类型众多,但无论什么电器,都是由开关控制的。而开关也有很多种,例如:拉链式开关、两位开关、调光开关 ......

对于开关和电器来说,不管任何时候,都可以在不触及另一方的情况下进行更换。比如,可以在不更换开关的情况下换掉灯泡(或风扇),也可以在不接触灯泡(或风扇)的情况下更换掉开关,甚至可以在不接触开关的情况下将灯泡和风扇互换。

这看起来很自然,当然也应该是这样!当不同的事物联系到一起时,它们应该在一个可以变更或者替换的系统中,以便不相互影响或者使影响尽可能的小,这样才能更方便、更低成本地去管理系统。试想一下,如果要更换房间里的一个灯泡,还必须把开关也换了,你会考虑使用这样的系统吗?

5

代码实现


创建实现类接口

所有电器都有一些共性,可以被打开和关闭:

// implementor.h #ifndef IMPLEMENTOR_H #define IMPLEMENTOR_H // 电器 class IEquipment { public: virtual ~IEquipment() {} // 打开 virtual void PowerOn() = 0; // 关闭 virtual void PowerOff() = 0;
}; #endif // IMPLEMENTOR_H 

创建具体实现类

接下来,是真正的电器 - 电灯和风扇,它们实现了 IEquipment 接口:

// concrete_implementor.h #ifndef CONCRETE_IMPLEMENTOR_H #define CONCRETE_IMPLEMENTOR_H #include "implementor.h" #include  // 电灯 class Light : public IEquipment
{ public: // 开灯 void PowerOn() override { std::cout << "Light is on." << std::endl;
    } // 关灯 void PowerOff() override { std::cout << "Light is off." << std::endl;
    }
}; // 风扇 class Fan : public IEquipment
{ public: // 打开风扇 void PowerOn() override { std::cout << "Fan is on." << std::endl;
    } // 关闭风扇 void PowerOff() override { std::cout << "Fan is off." << std::endl;
    }
}; #endif // CONCRETE_IMPLEMENTOR_H 

创建抽象类

对于开关来说,它并不知道电灯和风扇的存在,只知道自己可以控制(打开/关闭)某个电器。也就是说,每个 ISwitch 应该持有一个 IEquipment 对象:

// abstraction.h #ifndef ABSTRACTION_H #define ABSTRACTION_H #include "implementor.h" // 开关 class ISwitch { public:
    ISwitch(IEquipment *equipment) { m_pEquipment = equipment; } virtual ~ISwitch() {} // 打开电器 virtual void On() = 0; // 关闭电器 virtual void Off() = 0; protected:
    IEquipment *m_pEquipment;
}; #endif // ABSTRACTION_H 

创建扩充抽象类

特定类型的开关很多,比如拉链式开关、两位开关:

// refined_abstraction.h #ifndef REFINED_ABSTRACTION_H #define REFINED_ABSTRACTION_H #include "abstraction.h" #include  // 拉链式开关 class PullChainSwitch : public ISwitch
{ public:
    PullChainSwitch(IEquipment *equipment) 
        : ISwitch(equipment) {} // 用拉链式开关打开电器 void On() override { std::cout << "Switch on the equipment with a pull chain switch." << std::endl;
        m_pEquipment->PowerOn();
    } // 用拉链式开关关闭电器 void Off() override { std::cout << "Switch off the equipment with a pull chain switch." << std::endl;
        m_pEquipment->PowerOff();
    }
}; // 两位开关 class TwoPositionSwitch : public ISwitch
{ public:
    TwoPositionSwitch(IEquipment *equipment)
        : ISwitch(equipment) {} // 用两位开关打开电器 void On() override { std::cout << "Switch on the equipment with a two-position switch." << std::endl;
        m_pEquipment->PowerOn();
    } // 用两位开关关闭电器 void Off() override { std::cout << "Switch off the equipment with a two-position switch." << std::endl;
        m_pEquipment->PowerOff();
    }
}; #endif // REFINED_ABSTRACTION_H 

创建客户端

很好,是时候将开关和电器关联起来了:

// main.cpp #include "refined_abstraction.h" #include "concrete_implementor.h" #ifndef SAFE_DELETE #define SAFE_DELETE(p) { if(p){delete p; p=nullptr;} } #endif int main() { // 创建电器 - 电灯、风扇 IEquipment *light = new Light();
    IEquipment *fan = new Fan(); /**
    * 创建开关 - 拉链式开关、两位开关
    * 将拉链式开关和电灯关联起来,将两位开关和风扇关联起来
    **/ ISwitch *pullChain = new PullChainSwitch(light);
    ISwitch *twoPosition = new TwoPositionSwitch(fan); // 开灯、关灯 pullChain->On();
    pullChain->Off(); // 打开风扇、关闭风扇 twoPosition->On();
    twoPosition->Off();

    SAFE_DELETE(twoPosition);
    SAFE_DELETE(pullChain);
    SAFE_DELETE(fan);
    SAFE_DELETE(light);

    getchar(); return 0;
}

输出如下:

Switch on the equipment with a pull chain switch.

Light is on.

Switch off the equipment with a pull chain switch.

Light is off.

Switch on the equipment with a two-position switch.

Fan is on.

Switch off the equipment with a two-position switch.

Fan is off.



免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭