设计模式—结构型模式之桥接模式
将抽象与实现解耦,使两者都可以独立变化。
在现实生活中,某些类具有两个或多个维度的变化,如图形既可按形状分,又可按颜色分。如何设计类似于 Photoshop 这样的软件,能画不同形状和不同颜色的图形呢?如果用继承方式,m 种形状和 n 种颜色的图形就有 m×n 种,不但对应的子类很多,而且扩展困难。不同颜色和字体的文字、不同品牌和功率的汽车。
桥接将继承转为关联,降低类之间的耦合度,减少代码量。
桥接(Bridge)模式包含以下主要角色:
- 系统设计期间,如果这个类里面的一些东西,会扩展很多,这个东西就应该分离出来
- 抽象化(Abstraction)角色:定义抽象类,并包含一个对实现化对象的引用。
- 扩展抽象化(Refined Abstraction)角色:是抽象化角色的子类,实现父类中的业务方法,并通过组合关系调用实现化角色中的业务方法。
- 实现化(Implementor)角色:定义实现化角色的接口,供扩展抽象化角色调用。
- 具体实现化(Concrete Implementor)角色:给出实现化角色接口的具体实现。
举例
如果我们有不同型号的手机,每种型号的手机都有线下和线上两种销售渠道,两种渠道的价格还不同。如果我们不使用设计模式,我们就需要定义线上渠道的不同型号手机类、线下渠道的不同型号手机类;如果我们再增加一个渠道,又需要增加新的不同型号渠道手机类。我们能否把渠道分离出来呢?这便引出了桥接模式。
渠道抽象类:
/**
* 抽象渠道类
*/publicabstractclassAbstarctChnnel{privateString channel;privateInteger price;publicAbstarctChnnel(String channel,Integer price){this.channel = channel;this.price = price;}StringgetChannelInfo(){return"渠道:"+this.channel +"\t价格:"+this.price;}}
线上渠道和线下渠道分别为:
/**
* 线上渠道
*/publicclassOnlineChannelextendsAbstarctChnnel{publicOnlineChannel(String channel,Integer price){super(channel, price);}}/**
* 线下渠道
*/publicclassOfflineChannelextendsAbstarctChnnel{publicOfflineChannel(String channel,Integer price){super(channel, price);}}
我们的抽象手机类如下:
publicabstractclassAbstarctPhone{//桥接在此.....设计期间就得想好//【真正会引起此类变化的一个维度直接抽取出来,通过组合的方式接起来】//桥接+适配器 ...AbstarctChnnel chnnel;abstractStringgetPhone();publicvoidsetChnnel(AbstarctChnnel chnnel){this.chnnel = chnnel;}}
手机类如下:
publicclassBananaPhoneextendsAbstarctPhone{@OverrideStringgetPhone(){return"香蕉手机:"+this.chnnel.getChannelInfo();}}
测试类如下:
publicclassBridgeTest{publicstaticvoidmain(String[] args){BananaPhone phone =newBananaPhone();
phone.setChnnel(newOfflineChannel("线下渠道",10000));System.out.println("phone.getPhone() = "+ phone.getPhone());}}
运行结果如下:
如果我们再新增一个渠道,只要再扩展出一个渠道类即可,不需要新增手机类。
总结
我们需要把真正会引起此类变化的维度,直接抽取出来,通过组合的方式拼接起来。
版权归原作者 随机的未知 所有, 如有侵权,请联系我们删除。