09.桥接模式(结构型)

桥接模式(结构型)

一、桥接模式概述

桥接模式是一种很实用的结构型设计模式,如果软件系统中某个类存在两个独立变化的维度,通过该模式可以将这两个维度分离出来,使两者可以独立扩展,让系统更加符合“单一职责原则”。与多层继承方案不同,它将两个独立变化的维度设计为两个独立的继承等级结构,并且在抽象层建立一个抽象关联,该关联关系类似一条连接两个独立继承结构的桥,故名桥接模式。

1). 概念

  1. 桥接模式:将抽象部分与它的实现部分分离,使它们都可以独立地变化。它是一种对象结构型模式,又称为柄体(Handle and Body)模式或接口(Interface)模式。

桥接模式结构图.png

2). 相关角色

  1. Abstraction(抽象类):用于定义抽象类的接口,它一般是抽象类而不是接口,其中定义了一个Implementor(实现类接口)类型的对象并可以维护该对象,它与Implementor之间具有关联关系,它既可以包含抽象业务方法,也可以包含具体业务方法。

  2. RefinedAbstraction(扩充抽象类):扩充由Abstraction定义的接口,通常情况下它不再是抽象类而是具体类,它实现了在Abstraction中声明的抽象业务方法,在RefinedAbstraction中可以调用在Implementor中定义的业务方法。

  3. Implementor(实现类接口):定义实现类的接口,这个接口不一定要与Abstraction的接口完全一致,事实上这两个接口可以完全不同,一般而言,Implementor接口仅提供基本操作,而Abstraction定义的接口可能会做更多更复杂的操作。Implementor接口对这些基本操作进行了声明,而具体实现交给其子类。通过关联关系,在Abstraction中不仅拥有自己的方法,还可以调用到Implementor中定义的方法,使用关联关系来替代继承关系。

  4. ConcreteImplementor(具体实现类):具体实现Implementor接口,在不同的ConcreteImplementor中提供基本操作的不同实现,在程序运行时,ConcreteImplementor对象将替换其父类对象,提供给抽象类具体的业务操作方法。

桥接模式中提现了设计模式的“单一职责原则”、“开闭原则”、“合成复用原则”、“里氏代换原则”、“依赖倒转原则”等;通常情况下,我们将具有两个独立变化维度的类的一些普通业务方法和与之关系最密切的维度设计为“抽象类”层次结构(抽象部分),而将另一个维度设计为“实现类”层次结构(实现部分)

二、桥接模式案例(与适配器模式联用)

桥接模式用于系统的初步设计,将存在两个独立变化维度的类分为抽象化和实现化两个角色;完成初步设计后,发现系统与已有类无法协同工作,可以采用适配器模式;但是,有事在设计初期会用到大量第三方应用接口,需要使用适配器模式。

某报表处理系统,可以以多种方式展示报表,比如柱状图、折线图等,数据采集可以为数据库,可以为文本文件文件,甚至是Excel,但是使用Excel需要使用微软提供的第三方API库,而且没得源码,需要使用适配器模式。

报表.png
  1. Abstraction(抽象类):普通业务,业务逻辑关系比较密切
/**
 * 报表显示:提供多种显示方式(柱状图、折线图)
 *    角色:Abstraction 抽象类
 */
abstract class ReportForms {

    // 数据采集实现类
    protected DataGather dataGather;
    public void setDataGather(DataGather dataGather) {
        this.dataGather = dataGather;
    }

    // 展示数据
    public abstract void show();
}
  1. Implementor(实现类接口): 独立的业务功能,与抽象类业务处于不同的维度
/**
 * 数据采集:
 *   角色:Implementor(实现类接口)
 *   1. 支持多种数据库(以MySQL为例)
 *   2. 支持文本方式
 *   3. 支持Excel(结合 适配器模式:Target角色 )
 */
interface DataGather {
    public abstract String query();
}
  1. RefinedAbstraction(扩充抽象类)
/**
 * 柱状图
 *    角色:RefinedAbstraction 扩充抽象类
 */
class HistotramReportForms extends ReportForms {

    @Override
    public void show() {
        String data = this.dataGather.query();
        System.out.println("柱状图显示数据:" + data);
    }
}

/**
 * 折线图
 *    角色:RefinedAbstraction 扩充抽象类
 */
class LineChartReportForms extends ReportForms {

    @Override
    public void show() {
        String data = this.dataGather.query();
        System.out.println("折线图显示数据:" + data);
    }
}
  1. ConcreteImplementor(具体实现类)
/**
 * MySQL 数据库采集器
 *    角色: ConcreteImplementor 具体实现类
 */
class MySQLDataGather implements DataGather {

    @Override
    public String query() {
        return "来自MySQL的数据";
    }
}

/**
 * txt 文本数据采集器
 *    角色: ConcreteImplementor 具体实现类
 */
class TxtDataGather implements DataGather {

    @Override
    public String query() {
        return "来自文本文件的数据";
    }
}

/**
 * 适配器模式:来自Excel的数据
 *       adapter角色 + (桥接模式)ConcreteImplementor 具体实现类
 */
class ExcelDataGatherAdapter implements DataGather {

    private ExcelAPI excelAPI = new ExcelAPI();

    @Override
    public String query() {
        return this.excelAPI.getData();
    }
}

/**
 * 适配器模式:Excel API
 *      Adaptee角色
 */

class ExcelAPI {

    public String getData() {
        return "来自Excel文件的数据";
    }
}
  1. 客户端(建议使用xml + 反射,这里为了方便,直接new)
/**
 * 桥接模式 + 适配器模式
 * 建议使用XML配置 + 反射实现程序解耦合
 * @author Liucheng
 * @since 2019-07-21
 */
public class Client2 {
    public static void main(String[] args) {
        DataGather dataGather = new ExcelDataGatherAdapter();
        ReportForms reportForms = new HistotramReportForms();
        reportForms.setDataGather(dataGather);
        reportForms.show();
    }
}

三、桥接模式总结

1). 优点

  1. 分离抽象接口与实现部分:从而获得更多维度组合的对象。
  2. 极大减少子类的个数
  3. 可扩展性高,两个维度中任意扩展一个维度,对另一个没有影响,符合“开闭原则”

2). 缺点

  1. 难理解、设计有难度、要求设计者一开始就针对抽象层进行设计编程。
  2. 如何辨识不同的维度需要经验积累。

3). 适用场景

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

推荐阅读更多精彩内容

  • 【学习难度:★★★☆☆,使用频率:★★★☆☆】直接出处:桥接模式梳理和学习:https://github.com/...
    BruceOuyang阅读 899评论 0 2
  • 1.初识桥接模式 将抽象部分与它的实现部分分离,使它们都可以独立地变化。 Abstraction:抽象部分的接口。...
    王侦阅读 909评论 0 7
  • 点击进入我的博客 3.1 适配器模式 适配器模式把一个类的接口变换成客户端所期待的另一种接口,使得原本因接口不匹配...
    卢卡斯哔哔哔阅读 685评论 0 1
  • 这几年,我怕了,因为年龄越大,离开的亲人就越多,生命无常。 这几年,我变了,学会了让舍得的和舍不得的都随缘了。 这...
    斌斌有礼_2c23阅读 301评论 0 1
  • 最近看完了豆瓣推荐的“Call me by your name”,看的心里频起涟漪,每一段感情都值得被善待,这种感...
    小小小零一阅读 368评论 0 0