设计模式之桥接

来源:互联网 发布:网络女主播说唱歌曲 编辑:程序博客网 时间:2024/06/06 05:06

介绍

  • 实现与抽象分离, 使他们可以独立变化, 甚至运行时进行实现的切换;
  • 对象这个概念实际是由属性和行为两个部分组成的,属性我们可以认为是一种静止的,是一种抽象,一般情况下,行为是包含在一个对象中,但是,在有的情况下,我们需要将这些行为也进行归类,形成一个总的行为抽象,这就是桥模式;
  • 不希望抽象部分和行为有一种固定的绑定关系,而是应该可以动态联系的;

例子

  • 例如,一杯咖啡为例,子类实现类为四个:中杯加奶、大杯加奶、 中杯不加奶、大杯不加奶。
    但是,我们注意到:上面四个子类中有概念重叠,可从另外一个角度进行考虑,这四个类实际是两个角色的组合:抽象 和行为,其中抽象为:中杯和大杯;行为为:加奶 不加奶(如加橙汁 加苹果汁).
    实现四个子类在抽象和行为之间发生了固定的绑定关系,如果以后动态增加加葡萄汁的行为,就必须再增加两个类:中杯加葡萄汁和大杯加葡萄汁。显然混乱,扩展性极差。
  • 再入人, 可以具体分为男人, 女人, 人会有各种行为, 如工作, 学习, 娱乐…, 但对于不同职业的人所完成的具体行为行为内容是不同的, 显然如果为每个职业单独实现其行为是非常不好的设计,容易造成类的膨胀. 如下的结构中男人/女人继承与人, 男/女工人, 男/女教师, nan/女学生…又分别继承与男人/女人, 每个职业中都要实现人的必要行为: 吃饭, 睡觉, 学习, 工作, 游乐.., 这样非但有大量重复的代码不易扩展,如以后哪天要添加新的职业人, 又得同时实现男女连个类,或哪天出现了人妖, 哪必须为每个类加上人妖职业…

这里写图片描述

解决方案

  • 在我们每次使用设计模式时,我们应该根据设计原则去设计,而不是直接使用已有设计模式去套用,我们要明白的一点是设计模式是根据一定的设计原则而产生的。
    我们要遵循两个基本原则:
    • 找出变化封装之
    • 优先使用对象聚集,而不是继承
  • 前面的例子中, 变化的有抽象人的实现, 和职业的选择, 人完全可以选择不同的职业, 而不同的职业下会有不同的行为;所以我们可以将具体的人进行抽封装, 具体的职业进行抽象封装;
    如下图:

这里写图片描述

代码

// 抽象出变化的实现,可以用抽象工厂来选择具体的实现class Carrier{public:    virtual void vConHardWork() = 0;    virtual void vConLeisureWork() = 0;};class Teacher : public Carrier{public:    virtual void vConHardWork()    {        cout<<"teachers work hard for make more money!"<<endl;    }    virtual void vConLeisureWork()    {        cout<<"teachers work in a leisure-work company!"<<endl;    }};class Worker : public Carrier{public:    virtual void vConHardWork()    {        cout<<"workers work from 9:00am to 5:00pm everyday!"<<endl;    }    virtual void vConLeisureWork()    {        cout<<"workers work 8 hours in a leisure-company!"<<endl;    }};class Human{public:    Human(Carrier* ca,const string str):carir(ca),name(str){};    virtual void vHardWork(){};  // 不实现    virtual void vLeisureWork(){};protected:    string name;    Carrier* carir;};class MaleCarrier : public Human{public:    MaleCarrier(Carrier* ca,const string name):Human(ca,name){};    virtual void vHardWork()    {        cout<<"man need work hard for the family..."<<endl;        carir->vConHardWork();    }};class FemaleCarrier : public Human{public:    FemaleCarrier(Carrier* ca,const string name):Human(ca,name){};    virtual void vLeisureWork()    {        cout<<"female need pay more attention on care about family..."<<endl;        carir->vConLeisureWork();    }};
0 0