设计模式之桥接模式(Bridge)

一、初识桥接模式

对于电脑大家是在熟悉不过,看看大家的电脑,有dell的,有lenovo的。比如我的电脑室dell的,但是第一次装系统的时候用的是xp的系统,后来出 win7 了,于是我兴高采烈的装了一个win7旗舰版。关于电脑的分类的结构图大概是这样的。

设计模式之桥接模式(Bridge)_第1张图片

我们知道,电脑品牌多了去了,比如Acer,HP等等,系统也不只是XP和Win7,还有Linux和Unix等等啊,这样一来电脑的具体分类就多了去了。显然采用上面的结构图描述是不合理的了。我们看看还有木有其他的结构图,比如我们也可以这样的描述电脑的分类:

 

设计模式之桥接模式(Bridge)_第2张图片

先来分析一下上面的结构,将系统分为:XP系统和Win7系统,同时电脑按品牌分类可以分为Dell和Lenovo,然后电脑品牌和系统自由组合,这样就可以组合出XP系统的Dell、Win7系统的Dell、XP系统的Lenovo和Win7系统的Lenovo了。如果采用了上面的结构,现在要增加电脑品牌Acer和操作系统Linux只需要让Acer继承电脑,Linux继承系统,然后还是自由组合就行了。

实际上第二种结构就是采用的桥接模式画出的结构图了,GoF说:桥接模式(Bridge):将抽象部分与实现部分分离,使它们都可以独立的变化。什么叫抽象部分和实现部分分离?我们分析一下上面两种结构图,可以发现一个是用继承完成的,一种是用组合/聚合的方式完成的,而采用组合/聚合的方式就是所谓的抽象与实现分离。实际上在设计类时,我们应该首先考虑的是组合/聚合的方式,而不是考虑继承的方式,因为继承是一种强耦合关系,使用继承使得子类过多的依靠父类,这并不是很好。

二、桥接模式的代码实现

说了这么多,还是要亲自的实现以下桥接模式。不妨就实现这个电脑分类的例子吧:

#include <iostream>
 #include <string>
 
 using namespace std;
 
 
 class OS
 {
 private:
     string name;
 public:
     OS(string n):name(n){};
     void set_name(string n)
     {
         this->name = n;
     }
     string get_name()
     {
         return this->name;
     }
 };
 //XP操作系统
 class XP : public OS
 {
 public:
     XP(string n = "XP"):OS(n){};
 };
 //Win7操作系统
 class Win7 : public OS
 {
 public:
     Win7(string n = "Win7"):OS(n){};
 };
 //Linux操作系统
 class Linux : public OS
 {
 public:
     Linux(string n = "Linux"):OS(n){};
 };
 //抽象电脑品牌
 class Computer
 {
 private:
     string brand;
     OS os;
 public:
     Computer(OS xt,string b):brand(b),os(xt){};
     void set_os(OS os)
     {
         this->os = os;
     }
     OS get_os()
     {
         return this->os;
     }
     void init()
     {
         cout<<brand<<"电脑正在运行"<<os.get_name()<<endl;
     }
 };
 //Dell电脑
 class Dell : public Computer
 {
 public:
     Dell(OS os,string b = "Dell"):Computer(os,b){};
 };
 //Lenovo电脑
 class Lenovo : public Computer
 {
 public:
     Lenovo(OS os,string b = "Lenovo"):Computer(os,b){};
 };
 int main()
 {
     //实例化三个操作系统
     XP xp;
     Win7 win7;
     Linux linux;
     //建立两种品牌的电脑
     Dell dell(xp);
     Lenovo lenovo(win7);
     dell.init();
     lenovo.init();
     //我的dell重装linux
     dell.set_os(linux);
     dell.init();
     return 0;
 }



运行结果:

设计模式之桥接模式(Bridge)_第3张图片

小结:那么什么时候使用桥接模式呢?当系统可以从多个角度分类,每一种分类都有可能变化,那么就把这种多角度分类分离出来让他们独立变化,这样就可以减少他们之间的耦合。实际上使用桥接模式的好处已经不用多说了,在代码中和上面给出两种结构图中已经足够看出来了。那么想一想桥接模式和装饰模式有什么区别呢??桥接模式是适应多个维度变化的一种模式。而装饰模式是适应新需求的不断增加的一种模式。注意体会这两种模式的实现方式~~~

你可能感兴趣的:(设计模式之桥接模式(Bridge))