Android设计模式(二十三)-桥接模式

桥接模式也叫桥梁模式,和生活中一样,桥梁就是用来连接河道两岸的主要建筑。桥接模式也是起着连接两边的作用,连接的两边就是抽象部分和实现部分,这就需要在程序设计的时候划分好抽象部分和实现部分了。

定义

将抽象部分与实现部分分离,使他们都可以独立地进行变化。
博客地址

使用场景

  • 一个类存在两个独立维度的变化,且两个维度都需要进行拓展。
  • 一个系统需要在构件的抽象化角色和具体化角色之间增加更多的灵活性,避免两个层次之间建立静态的继承联系,可以用桥接模式使他们在抽象层建立一个关联关系。
  • 不想使用继承或使用继承会导致生成一大堆类的时候。

UML

  • Abstraction: 抽象部分,抽象部分保持对实现部分的引用,抽象部分的方法要调用实现部分的对象来实现。一般为抽象类。
  • RefinedAbstraction: 优化的抽象部分,一般是对抽象部分的方法进行完善和拓展,是抽象部分的具体实现
  • Implementor:实现部分,可以是接口和抽象类,方法不一定要和抽象部分保持一致。一般情况下由实现部分提供基本的操作,而抽象部分定义基于这些操作的业务方法。
  • ConcreteImplementorA,B: 具体的实现部分。

模板代码:

实现部分的接口:

public interface Implementor{
    void operationImpl();
}

具体的实现部分:

public class ConcreteImplementorA implements  Implementor{

    @Override
    public void operationImpl() {
        //具体的实现
    }
}
public class ConcreteImplementor implements  Implementor{

    @Override
    public void operationImpl() {
        //具体的实现
    }
}

抽象部分:

public abstract class Abstraction{
    private Implementor mImplementor;

    public Abstraction(Implementor mImplementor) {
        this.mImplementor = mImplementor;
    }
    public void operation(){
        //调用实现部分的具体方法
        mImplementor.operationImpl();
    }
}

优化的抽象部分:

public class RefinedAbstraction extends Abstraction{

    public RefinedAbstraction(Implementor mImplementor) {
        super(mImplementor);
    }
    //可以增加拓展其他方法,也可以重写父类的方法,也能调用父类的方法

    @Override
    public void refinedperation() {
        //对抽象的父类的方法进行拓展
    }
}

简单实现

拿书中举得例子。这里用桥接模式来建立两个维度之间的联系。对咖啡来说,可以分为两个维度,杯子大小是一个维度,加不加糖又是一个维度。这两个没有谁是抽象部分谁是具体部分。就拿杯子大小作为抽象部分来做一个简单实现:

抽象的糖,相当于实现部分的接口:

public abstract class CoffeeSugar {
    public abstract void makeSugar();
}

两种实现部分的实现,加不加糖:

public class AddSugar extends CoffeeSugar {
    @Override
    public void makeSugar() {
        System.out.println("加糖的");
    }
}
public class NoSugar extends CoffeeSugar {
    @Override
    public void makeSugar() {
        System.out.println("不加糖的");
    }
}

杯子的抽象,相当于抽象部分,持有一个糖的引用:

public abstract class CoffeeCup {
    protected CoffeeSugar coffeeSugar;

    public CoffeeCup(CoffeeSugar coffeeSugar) {
        this.coffeeSugar = coffeeSugar;
    }

    public  void makeCup(){
        coffeeSugar.makeSugar();
    }
}

优化的抽象部分,有大小两种杯子:

public class LargeCup extends CoffeeCup {
    public LargeCup(CoffeeSugar coffeeSugar) {
        super(coffeeSugar);
    }

    @Override
    public void makeCup() {
        System.out.println("大杯的");
        super.makeCup();
    }
}
public class SmallCup extends CoffeeCup {
    public SmallCup(CoffeeSugar coffeeSugar) {
        super(coffeeSugar);
    }

    @Override
    public void makeCup() {
        System.out.println("小杯的");
        super.makeCup();
    }
}

客户端调用:

public class Client {
    public static void main(String[] args) {
        CoffeeSugar addSugar = new AddSugar();
        CoffeeSugar noSugar = new NoSugar();
        CoffeeCup coffee = new LargeCup(addSugar);
        coffee.makeCup();
        System.out.println("---");
        coffee = new LargeCup(noSugar);
        coffee.makeCup();
        System.out.println("---");
        coffee = new SmallCup(noSugar);
        coffee.makeCup();
    }
}

输出:

这样就把两个维度连接到一起了。而且两个维度是可以独立拓展的。比如如果想加上个中杯,或者来个多糖少糖等分类,只需要多实现几个类就行了,然后由客户端去调用。这样就能在两个维度上独立的拓展。

再想加上第三个维度也是很简单的,现在要加上年龄分类,有老年人喝的喝年轻人喝的,者又是一个维度。

这时可以把前面两个已经连接在一起的看做是一个维度,让新的去桥接者个已有的。

抽象的people,持有一个之前的桥接,把people和CoffeeCup连接起来:

ublic abstract class People {
    protected CoffeeCup coffeeCup;

    public People(CoffeeCup coffeeCup) {
        this.coffeeCup = coffeeCup;
    }

    public abstract void age();
}

具体的年龄分类:

public class YoungPeople extends People {
    public YoungPeople(CoffeeCup coffeeCup) {
        super(coffeeCup);
    }

    @Override
    public void age() {
        System.out.println("年轻人喝的");
        coffeeCup.makeCup();
    }
}
public class OldPeople extends People {
    public OldPeople(CoffeeCup coffeeCup) {
        super(coffeeCup);
    }

    @Override
    public void age() {
        System.out.println("老年人喝的");
        coffeeCup.makeCup();
    }
}

然后客户端只需要这样

public class Client {
    public static void main(String[] args) {
        CoffeeSugar addSugar = new AddSugar();
        CoffeeSugar noSugar = new NoSugar();
        CoffeeCup coffee = new LargeCup(addSugar);
        coffee.makeCup();
        System.out.println("---");
        coffee = new LargeCup(noSugar);
        coffee.makeCup();
        System.out.println("---");
        coffee = new SmallCup(noSugar);
        coffee.makeCup();
        System.out.println("-----");
        //再次桥接
        People people = new OldPeople(coffee);
        people.age();
    }
}

输出:

总结

桥接模式就是把系统分为抽象部分和实现部分,而建立桥接的方式也很简单。就是让抽象部分持有实现部分的引用,可以通过这个引用调用实现部分的具体方法。

使用这个系统最重要的是把握系统的分离,分不好就失去了灵活的拓展性,因此不容易设计。

优点

  • 分离成抽象部分和实现部分,并且两部分都可以独立的拓展,一个部分变化不会引起另一部分的变化,提高了系统的拓展性。
  • 复用性强,避免了使用继承产生大量继承类的问题。

缺点

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

推荐阅读更多精彩内容