桥接(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了。后来牧师又可以拿剑和魔法书了,估计开发人员要一头撞死了。。。

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

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

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

如果一个系统需要在构件的抽象化角色和具体化角色之间增加更多的灵活性,避免在两个层次之间建立静态的继承联系,通过桥接模式可以使它们在抽象层建立一个关联关系。

对于那些不希望使用继承或因为多层次继承导致系统类的个数急剧增加的系统,桥接模式尤为适用。

一个类存在两个独立变化的维度,且这两个维度都需要进行扩展,就是M x N问题,让M和N自己玩儿。

java设计模式 桥接_Java设计模式百例 - 桥接模式相关推荐

  1. java单例设计模式双重_Java 设计模式 ——单例模式(饿汉,懒汉,双重锁,静态内部类)...

    设计模式: 是在大量的实践中总结和理论化之后优选的代码结构,编程风格,以及解决问题的思考方式.设计模式免去我们自己再思考和摸索.就像是经典的棋谱,不同的棋局,我们用不同的棋谱 俗称"套路&q ...

  2. java 设计模式 示例_Java设计模式–示例教程

    java 设计模式 示例 Design Patterns are very popular among software developers. A design pattern is a well- ...

  3. java 影子实例_java设计模式(四)--单例模式

    Singleton最熟悉不过了,下面学习单例模式.转载:http://zz563143188.iteye.com/blog/1847029 单例对象(Singleton)是一种常用的设计模式.在Jav ...

  4. java单例设计模式懒汉_Java设计模式之单例设计模式(懒汉、饿汉)

    [toc] Java设计模式之单例设计模式(懒汉.饿汉) 相信面试过的初中级Java开发的朋友可能都有遇到过单例设计模式的笔试题吧,如果之前没有背下来或者不理解,可以看看下面这篇文章,应该足够应付笔试 ...

  5. java设计模式教程_Java设计模式教程

    java设计模式教程 课程大纲 架构和计算机科学中的设计模式是记录特定专业领域中设计问题的解决方案的正式方法. 这个想法是由建筑师Christopher Alexander在建筑领域引入的,并已被修改 ...

  6. java工厂方法_Java设计模式之工厂方法模式

    2.工厂方法模式 <设计模式之禅>中用神话故事女娲造人来比概述工厂方法模式. 女娲采集黄土捏成人的形状,然后放到八卦炉中烧制,最后放置到大地上生长,工艺过程是没有错的,但是意外随时都会发生 ...

  7. java外观设计修改_Java设计模式之外观模式和装饰器模式的设计(精选)

    前言 本篇来学习下结构型模式的外观模式和装饰器模式. 外观模式 简介 外观模式隐藏系统的复杂性,并向客户端提供了一个客户端可以访问系统的接口.这种类型的设计模式属于结构型模式,它向现有的系统添加一个接 ...

  8. java 对象工厂_Java设计模式之--工厂方式

    在分析工厂模式之前,我们先看看普通模式下的Java三层结构. Controller(表现层)-Service(业务层)-Dao(持久层.数据库连接层),基于此三层结构完成JavaWeb相关业务. 假设 ...

  9. java 设计模式原则_Java设计模式的七大原则

    设计原则名称 定  义 使用频率 单一职责原则(Single Responsibility Principle, SRP) 一个类只负责一个功能领域中的相应职责. ★★★★☆ 开闭原则(Open-Cl ...

最新文章

  1. Java窗口(JFrame)从零开始(8)——文本框+文本域+密码框
  2. 操作系统原理:进程与线程、进程生命周期、线程的类型
  3. 从零搭建 dubbogo
  4. WebStorm 常用功能的使用技巧分享
  5. ssh-keygen -t rsa执行后提示没有权限
  6. to_string作用
  7. 谷歌地球最新host_听说《流浪地球》被豆瓣鄙视了,我们用数据看一把真相(上)...
  8. mysql1045错误解读_谈谈MYSQL ERROR 1045 错误的解决办法!
  9. springloaded热加载
  10. neo4j 学习笔记
  11. 【转】vc++MCF/C++/C中怎样让应用程序获得或以管理员权限运行 ,ShellExecuteEX编程 --- 获取管理员权限...
  12. 【Hinton论文翻译与理解】How to represent part-whole hierarchies in a neural network_202102
  13. Linux 火狐无法播放视频
  14. 呼叫中心点击拨打接口升级代码
  15. OFDM载波间隔_NBloT上行子载波
  16. Elasticsearch 谷歌插件 Elasticsearch-head 使用
  17. Word中序号后面有空格怎么删除?
  18. JS 四舍六入五成双
  19. Windows日志分析(中)
  20. 11. python入门复习教程之命名空间与作用域,再谈异常,标准库,持久化与序列化

热门文章

  1. 动态获取图片尺寸-uniapp
  2. java下标_Java 在Word中设置上标、下标
  3. 第三季百度网盘AI大赛盛夏来袭,寻找热爱AI的你!
  4. VGA/DVI/HDMI/DP/Type-C等常用显示接口对比介绍
  5. 问:发电机运行中的监视和检查的主要内容有哪些?
  6. 山海演武传·黄道·第一卷 雏龙惊蛰 第二十 ~ 二十一章 异世界的算命大师
  7. 【Houdini】Houdini实现Realflow大脑袋案例
  8. 不用网线 一样部署 无线宽带电力猫产品
  9. 诺大一个中国,却无书可读
  10. c++ 报错 error: expected unqualified-id before numeric constant