Java设计模式百例 - 桥接模式

本文源码见:https://github.com/get-set/get-designpatterns/tree/master/bridge

桥接(Bridge)是用于把抽象化与实现化解耦,使得二者可以独立变化。这种类型的设计模式属于结构型模式,它通过提供抽象化和实现化之间的桥接结构,来实现二者的解耦。这种模式涉及到一个作为桥接的接口,使得实体类的功能独立于接口实现类。这两种类型的类可被结构化改变而互不影响。

上面这段是抄来的,读下来还是有点懵逼,不过并不是没用,我们还是老规矩,先从一个例子入手,然后回头再来回味这段话,就会有更深入的理解。

例子

相信大家都玩过或正在玩网游,不知道有没有玩过《魔兽世界》,啊~那是我逝去的青春啊~想当年,我是个快乐的,内心兼备光明与黑暗的牧师。当我使用“神圣”天赋的时候,我是一名救死扶伤的奶妈,给队友加血;当我使用“暗影”天赋的时候,我是一名嫉恶如仇的战士,给敌人伤害。

牧师可以使用多种武器,比如锤子和魔杖。一把锤子或魔杖在切换不同天赋的时候都能使用,在“神圣”天赋的时候,武器具备增加治疗的属性,使用的时候发出金黄色的光芒,可以给队友和自己加血;在“暗影”天赋的时候,武器具备增加伤害的属性,使用的时候发出黑紫色的光芒,可以对敌人造成伤害。

无论使用什么武器,都需要施法,施法前要吟唱,然后挥舞一下武器放出法术,最后收回。吟唱的时候微微发光,放出法术的时候光芒从武器到目标任务,收回的时候光芒消失。那么对于法术来说如下设计:

Enchantment.java

public interface Enchantment {
    void onActivate();
    void apply();
    void onDeactivate();
}

HolyEnchantment.java

public class HolyEnchantment implements Enchantment {
    public void onActivate() {
        System.out.println("武器逐渐泛起金黄色的圣光...");
    }

    public void apply() {
        System.out.println("一道金黄色的圣光从武器发出,传到队友身上,队友加血1000");
    }

    public void onDeactivate() {
        System.out.println("武器的光芒迅速消失...");
    }
    
    @Override
    public String toString() {
        return "神圣魔法";
    }
}

ShadowEnchantment.java

public class ShadowEnchantment implements Enchantment {
    public void onActivate() {
        System.out.println("武器逐渐泛起黑紫色的暗影...");
    }

    public void apply() {
        System.out.println("一道黑紫色的暗影从武器发出,传到敌人身上,敌人掉血2000");
    }

    public void onDeactivate() {
        System.out.println("武器的暗影迅速消失...");
    }
    
    @Override
    public String toString() {
        return "暗影魔法";
    }
}

对于武器来说,进行如下设计:

Weapon.java

public abstract class Weapon {
    protected Enchantment enchantment;
    abstract void chant();
    abstract void wield();
    abstract void retrieve();
    public void setEnchantment(Enchantment enchantment) {
        this.enchantment = enchantment;
    }
    public Enchantment getEnchantment() {
        return this.enchantment;
    }
}

Hammer.java

public class Hammer extends Weapon {

    public Hammer(Enchantment enchantment) {
        this.enchantment = enchantment;
    }

    public void chant() {
        System.out.print("牧师拿出锤子,口中吟唱神圣治疗祷言...");
        enchantment.onActivate();
    }

    public void wield() {
        System.out.print("牧师将锤子举过头顶挥舞了一下...");
        enchantment.apply();
    }

    public void retrieve() {
        System.out.print("牧师收回锤子...");
        enchantment.onDeactivate();
    }
}

Wand.java

public class Wand extends Weapon {
    public Wand(Enchantment enchantment) {
        this.enchantment = enchantment;
    }

    public void chant() {
        System.out.print("牧师拿出魔杖,口中吟唱神圣治疗祷言...");
        enchantment.onActivate();
    }

    public void wield() {
        System.out.print("牧师将魔杖举过头顶挥舞了一下...");
        enchantment.apply();
    }

    public void retrieve() {
        System.out.print("牧师收回魔杖...");
        enchantment.onDeactivate();
    }
}

测试一下效果:

Client.java

public class Client {
    public static void main(String[] args) {
        System.out.println("进入副本,使用神圣天赋 >>>");
        Enchantment enchantment = new HolyEnchantment();
        Wand wand = new Wand(enchantment);
        wand.chant();
        wand.wield();
        wand.retrieve();

        System.out.println("野外打怪,使用暗影天赋 >>>");
        enchantment = new ShadowEnchantment();
        Hammer hammer = new Hammer(enchantment);
        hammer.chant();
        hammer.wield();
        hammer.retrieve();
    }
}

输出如下:

进入副本,使用神圣天赋 >>>
牧师拿出魔杖,口中吟唱神圣治疗祷言...武器逐渐泛起金黄色的圣光...
牧师将魔杖举过头顶挥舞了一下...一道金黄色的圣光从武器发出,传到队友身上,队友加血1000
牧师收回魔杖...武器的光芒迅速消失...
野外打怪,使用暗影天赋 >>>
牧师拿出锤子,口中吟唱神圣治疗祷言...武器逐渐泛起黑紫色的暗影...
牧师将锤子举过头顶挥舞了一下...一道黑紫色的暗影从武器发出,传到敌人身上,敌人掉血2000
牧师收回锤子...武器的暗影迅速消失...

总结

我们再来看一下文章开头的话“把抽象化与实现化解耦”。在理解这句话前,我们先看看本例为什么这么设计。

本例在设计的时候,将魔法作为武器的成员进行了组合。如果不使用组合这种方式呢?先不考虑Java是否能多继承,对于一把武器一次特定的使用,需要继承某种Weapon和某种Enchantment,这样就有了“神圣锤子”、“暗影锤子”、“神圣魔杖”和“暗影魔杖”。

其实我们这里遇到的是一个M x N的问题。对于一把武器来说,可能是锤子可能是魔杖,而且比如有魔法才能使用,可能是神圣魔法,可能是暗影魔法,可见这是个2 x 2的问题。

那么如果游戏可玩性增加,牧师又多了个天赋(实际上牧师还有个“戒律”天赋),这个时候需要增加“戒律锤子”和“戒律魔杖”,现在是2 x 3了。后来牧师又可以拿剑和魔法书了,估计开发人员要一头撞死了。。。

实现系统可能有多个角度分类,每一种角度都可能变化,这里武器类型和天赋类型就是两个角度,它们分别会有变化。这两个角度都有不同的抽象(比如武器和天赋)和实现(比如锤子、魔杖;神圣、暗影)。把这种多角度分类给分离出来让他们独立变化,减少他们之间耦合,就能够有效提高扩展性,如何做到呢?

就如同本例的方式,使用抽象化和实现化之间使用关联关系(组合或者聚合关系)而不是继承关系,从而使两者可以相对独立地变化,这就是桥接模式的用意。所以解耦就是将抽象化和实现化之间的耦合解脱开,或者说是将它们之间的强关联(可以理解成编译时确定的关联)改换成弱关联(可以理解成运行时才确定的关联),将两个角色之间的继承关系改为关联关系。

所以,桥接模式的使用场景如下:

  1. 如果一个系统需要在构件的抽象化角色和具体化角色之间增加更多的灵活性,避免在两个层次之间建立静态的继承联系,通过桥接模式可以使它们在抽象层建立一个关联关系。
  2. 对于那些不希望使用继承或因为多层次继承导致系统类的个数急剧增加的系统,桥接模式尤为适用。
  3. 一个类存在两个独立变化的维度,且这两个维度都需要进行扩展,就是M x N问题,让M和N自己玩儿。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容

  • 1 场景问题# 1.1 发送提示消息## 考虑这样一个实际的业务功能:发送提示消息。基本上所有带业务流程处理的系统...
    七寸知架构阅读 5,000评论 5 63
  • 在正式介绍桥接模式之前,我先跟大家谈谈两种常见文具的区别,它们是毛笔和蜡笔。假如我们需要大中小3种型号的画笔,能够...
    justCode_阅读 1,767评论 0 7
  • 原文链接:http://blog.csdn.net/zhangerqing http://www.cnblogs....
    孤独杂货铺阅读 1,513评论 0 3
  • 是“功夫不负有心人”呢,还是对某些事过于执着了。近两个月的苦苦寻觅,终于让我找到了这件看上去相似度百分之九十的宝贝...
    潘玥吉阅读 171评论 0 0
  • 我总觉得我该写点儿什么。可忙碌、繁杂、懒惰这些占据了大部分的时间,让我无暇顾及,无法思考,无法及时的把我...
    北在北方3582阅读 191评论 2 0