结构型模式-桥接模式

情景:由于产品线的壮大,现考虑将当前UI库扩展到其他平台,即一个可移植的Window抽象部分的实现,这一抽象部分应该允许用户开发一些在X Window System和IBM的Presentation Manager(PM)系统中都可以使用的窗体应用程序(Target)。

分析:看起来似乎不简单,于是我们立马着手开干。首先运用继承机制,我们可以定义Window抽象类和它的两个子类XWindow和PMWindow,由它们分别实现不同系统平台上的Window界面,事情发展得不错~

问题:当我们准备使Window的一个子类IconWindow(用于图标处理)支持两个系统平台时,我们发现必须实现两个新类XIconWindow和PMIconWindow,更为糟糕的是,我们必须对每一种类型的窗口都要定义两个类,考虑到后面还可能会支持其他平台,每增加一个平台都需要对类型窗口定义新的Window子类,另外,继承机制使得客户代码与平台相关,每当客户创建一个窗口,都必须要实例化一个具体的类,这个类有特定的实现方式,这将使得很难将客户代码移植到其他平台上去。换句话来说,创建窗口时不应该涉及具体的实现部分(不同平台会有不同),应该仅仅让窗口的实现部分依赖于应用运行的平台,创建窗口这个过程不应涉及特定平台。

解决方案:使用桥接模式,即将Window抽象和它的实现部分分别放在独立的类层次结构中,其中一个类层次结构针对窗口接口(Window、IconWindow)另外一个独立的类层次结构针对平台相关的窗口实现部分(绘制、布局),这个类层次结构的根类为WindowImp(如XWindowImp提供了一个基于X Window系统的实现),Window子类的所有操作都是用WindowImp接口中的抽象操作实现的。这就将窗口的抽象与系统平台相关的实现部分分离开来。我们将Window与WindowImp之间的关系称之为桥接,因为它在抽象类与它的实现之间起到了桥梁的作用。

桥接模式.png

关键字:
Abstraction(Window)--定义抽象类的接口,维护一个指向Implementor类型对象的指针

RefinedAbstraction(IconWindow)--扩充由Abstraction定义的接口

Implementor(WindowImp)--定义实现类的接口,Implementor接口提供基本操作,而Abstraction则定义了基于这些基本操作的较高层次的操作

ConcreteImplementor(XWindowImp)--实现Implementor接口并定义它的具体实现

实现:

struct Point
{
public:
    Point(int x, int y) :m_iX(x), m_iY(y) {};

    int X() const { return m_iX; }
    int Y() const { return m_iY; }

private:
    int m_iX;
    int m_iY;
};

class View
{
public:
    void DrawOn(Window *);

};

class WindowImp;

class Window
{
public:
    Window(View *content);

    virtual void DrawContents();

    virtual void Open();
    virtual void Close();
    virtual void Iconify();
    virtual void Deiconify();

    virtual void SetOrigin(const Point &at);
    virtual void SetExtent(const Point &extent);
    virtual void Raise();
    virtual void Lower();
    virtual void DrawLine(const Point &, const Point &);
    virtual void DrawRect(const Point &, const Point &);
    virtual void DrawPolygon(const Point *,int n);
    virtual void DrawText(const char *, const Point &);

protected:
    WindowImp* GetWindowImp();
    View* GetView();

private:
    WindowImp *m_pWinImp;
    View *m_pView;
};

void Window::DrawRect(const Point &p1, const Point &p2)
{
    WindowImp *imp = GetWindowImp();
    if (imp != nullptr)
    {
        imp->DeviceRect(p1.X(), p1.Y(), p1.X(), p1.Y());
    }
}

void Window::DrawText(const char *text, const Point &p1)
{
    WindowImp *imp = GetWindowImp();
    if (imp != nullptr)
    {
        imp->DeviceText(text, p1.X(), p1.Y());
    }
}

class ApplicationWindow : public Window
{
public:
    virtual void DrawContents();
};

void ApplicationWindow::DrawContents()
{
    GetView()->DrawOn(this);
}

class IconWindow : public Window
{
public:
    virtual void DrawContents();
private:
    const char *m_bitmapName;
};

void IconWindow::DrawContents()
{
    WindowImp *imp = GetWindowImp();
    if (imp != nullptr)
    {
        imp->DeviceBitmap(m_bitmapName, 0, 0);
    }
}

class  WindowImp
{
public:
    virtual void ImpTop() = 0;
    virtual void ImpBottom() = 0;
    virtual void ImpSetExtent(const Point &) = 0;
    virtual void ImpSetExtent(const Point &) = 0;

    virtual void DeviceRect(int, int, int, int) = 0;
    virtual void DeviceText(const char *, int, int) = 0;
    virtual void DeviceBitmap(const char *, int, int) = 0;

protected:
    WindowImp();

};

class Display;
class GC;
void XDrawRectangle(Display *display, int iWindowId, GC* gc, int x, int y, int width, int height);
class XWindowImp : public WindowImp
{
public:
    XWindowImp();
    virtual void DeviceRect(int, int, int, int);

private:
    //system-specific state
    Display *m_pDisplay;
    int m_iWindowId;    //window id
    GC *m_pGc;          //window graphic context
};

void XWindowImp::DeviceRect(int x0, int y0, int x1, int y1)
{
    int x = ::round(std::min(x0, x1));
    int y = ::round(std::min(y0, y1));
    int w = ::round(std::abs(x0 - x1));
    int h = ::round(std::abs(y0 - y1));
    XDrawRectangle(m_pDisplay, m_iWindowId, m_pGc, x, y, w, h);
}

class HPS;
class PMWindowImp : public WindowImp
{
public:
    PMWindowImp();
    virtual void DeviceRect(int, int, int, int);

private:
    //system-specific state
    HPS *m_pHps;            
};

void PMWindowImp::DeviceRect(int x0, int y0, int x1, int y1)
{
    int left = std::min(x0, x1);
    int right = std::max(x0, x1);
    int bottom = std::min(y0, y1);
    int top = std::max(y0, y1);

    Point points[4] = 
    {
        Point (left,top),
        Point(right,top), 
        Point(right,bottom), 
        Point(left,bottom)
    };

    // to do draw graphic...
}

WindowImp* Window::GetWindowImp()
{
    if (m_pWinImp == nullptr)
    {
        m_pWinImp = WindowSystemFactor::Instance()->MakeWindowImp();
    }
    return m_pWinImp;
}

class WindowSystemFactor
{
public:
    static WindowSystemFactor *Instance();
    WindowImp *MakeWindowImp();

protected:
    WindowSystemFactor();
};

适用性:

  • 你不希望在抽象和它的实现部分之间有一个固定的绑定关系。例如,在程序运行时刻实现部分应可以被选择或者切换
  • 类的抽象以及它的实现都应该可以通过生成子类的方法加以扩充,Bridge模式可以对不同的抽象接口和实现部分进行组合,并对他们进行扩充。
  • 对一个抽象的实现部分的修改应对客户不产生影响,即客户的代码不必重新编译。

应用:
1.libg++类库定义了一些类用于实现公共的数据结构,例如Set、LinkedSet、HashSet、LinkedList和HashTable。Set是一个抽象类,它定义了一组抽象接口,而LinkedList和HashTable则分别是连表和hash表的具体实现,LinkedSet和HashSet是Set的实现者,它们桥接了Set和它们具体所对应LinkedList和HashTable。
2.Android View视图层级中的CheckBox、Button、TextView和View之间构成一个继承关系的视图层级,每一个视图定义了该类控件所拥有的基本属性和行为,但正真绘制到屏幕上的部分是由与View相关的功能实现类DisplayList、HardwareLayer、Canvas负责,这两部分的关系可以看作是对桥接模式的应用。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,222评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,455评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,720评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,568评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,696评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,879评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,028评论 3 409
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,773评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,220评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,550评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,697评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,360评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,002评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,782评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,010评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,433评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,587评论 2 350