Java設(shè)計(jì)模式中橋接模式應(yīng)用詳解
編程是一門(mén)藝術(shù),大批量的改動(dòng)顯然是非常丑陋的做法,用心的琢磨寫(xiě)的代碼讓它變的更美觀。
顯示生活中有物品具有多個(gè)維度的屬性,比如文字有顏色和字體兩個(gè)維度的差別,汽車(chē)有顏色和牌子兩個(gè)維度的屬性等。如果用繼承方式,不但對(duì)應(yīng)的子類(lèi)很多,而且擴(kuò)展困難。如果用橋接模式就能很好地解決這些問(wèn)題。
1.橋接(Bridge)設(shè)計(jì)模式定義
橋接模式是:將抽象與實(shí)現(xiàn)分離,使它們可以獨(dú)立變化。它是用組合關(guān)系代替繼承關(guān)系來(lái)實(shí)現(xiàn),從而降低了抽象和實(shí)現(xiàn)這兩個(gè)可變維度的耦合度。
橋接模式遵循了里氏替換原則和依賴倒置原則,最終實(shí)現(xiàn)了開(kāi)閉原則,對(duì)修改關(guān)閉,對(duì)擴(kuò)展開(kāi)放。
2.橋接設(shè)計(jì)模式優(yōu)點(diǎn)以及不足
橋接(Bridge)模式的優(yōu)點(diǎn)是:
- 抽象與實(shí)現(xiàn)分離,擴(kuò)展能力強(qiáng)
- 符合開(kāi)閉原則
- 符合合成復(fù)用原則
- 其實(shí)現(xiàn)細(xì)節(jié)對(duì)客戶透明
不足是:由于聚合關(guān)系建立在抽象層,要求開(kāi)發(fā)者針對(duì)抽象化進(jìn)行設(shè)計(jì)與編程,能正確地識(shí)別出系統(tǒng)中兩個(gè)獨(dú)立變化的維度,這增加了系統(tǒng)的理解與設(shè)計(jì)難度。
3.橋接設(shè)計(jì)的實(shí)現(xiàn)思路
可以將抽象化部分與實(shí)現(xiàn)化部分分開(kāi),取消二者的繼承關(guān)系,改用組合關(guān)系。
橋接設(shè)計(jì)模式的結(jié)構(gòu):
- 抽象化(Abstraction)角色:定義抽象類(lèi),并包含一個(gè)對(duì)實(shí)現(xiàn)化對(duì)象的引用。
- 擴(kuò)展抽象化(Refined Abstraction)角色:是抽象化角色的子類(lèi),實(shí)現(xiàn)父類(lèi)中的業(yè)務(wù)方法,并通過(guò)組合關(guān)系調(diào)用實(shí)現(xiàn)化角色中的業(yè)務(wù)方法。
- 實(shí)現(xiàn)化(Implementor)角色:定義實(shí)現(xiàn)化角色的接口,供擴(kuò)展抽象化角色調(diào)用。
- 具體實(shí)現(xiàn)化(Concrete Implementor)角色:給出實(shí)現(xiàn)化角色接口的具體實(shí)現(xiàn)。
4.橋接設(shè)計(jì)模式示例
示例場(chǎng)景分析:拿袋子裝采摘的水果,袋子根據(jù)容量有大、中、小三個(gè)類(lèi)別,袋子根據(jù)材質(zhì)有塑料、紙質(zhì)、麻袋三個(gè)類(lèi)別,下面使用橋接的模式進(jìn)行設(shè)計(jì):
/** * 采摘容器 */ public abstract class BagAbstraction { /** * 這個(gè)是容器的另外一組屬性 */ protected Material material; /** * 另外一組屬性,通過(guò)set方法注入進(jìn)來(lái) * @param material */ public void setMaterial(Material material){ this.material = material; } /** * 采摘 */ public abstract void pick(); } public class BigBag extends BagAbstraction { public void pick(){ this.material.draw(); System.out.println("采摘了一大袋"); } } public class MidBag extends BagAbstraction { public void pick(){ this.material.draw(); System.out.println("采摘了一中型袋"); } } public class SmallBag extends BagAbstraction { public void pick(){ this.material.draw(); System.out.println("采摘了一小袋"); } } //材質(zhì)接口 public interface Material { public void draw(); } public class Paper implements Material{ public void draw(){ System.out.print("用紙盒"); } } public class Plastic implements Material{ public void draw(){ System.out.print("用塑料袋"); } } public class Sack implements Material{ public void draw(){ System.out.print("用麻袋"); } } public class BridgeTest { public static void main(String[] args) { /*袋子型號(hào)*/ BagAbstraction bag = new SmallBag(); /*袋子材質(zhì)*/ Material material = new Paper(); bag.setMaterial(material); /*開(kāi)始采摘*/ bag.pick(); } }
在抽象類(lèi)BagAbstraction里通過(guò)組合的方式跟材質(zhì)建立聯(lián)系,通過(guò)set方法注入,這樣就將第二個(gè)維度的屬性添加進(jìn)來(lái)了,這樣就將物品的兩個(gè)維度的屬性進(jìn)行分離,這樣清晰代碼可讀性強(qiáng),并且方便進(jìn)行擴(kuò)展,即便是再多一個(gè)維度也方便進(jìn)行關(guān)聯(lián)。
5.橋接模式的應(yīng)用場(chǎng)景
橋接設(shè)計(jì)模式一般用在:產(chǎn)品具有多維度的時(shí)候,通過(guò)繼承的方式子類(lèi)非常多,可以采用橋接的方式,將商品的多個(gè)維度進(jìn)行分離設(shè)計(jì)。這樣方便擴(kuò)展,代碼可讀性強(qiáng),可以解耦這些變化的維度,使高層代碼架構(gòu)穩(wěn)定。
當(dāng)一個(gè)類(lèi)內(nèi)部具備兩種或多種變化維度時(shí),使用橋接模式
- 當(dāng)一個(gè)類(lèi)存在兩個(gè)獨(dú)立變化的維度,且這兩個(gè)維度都需要進(jìn)行擴(kuò)展時(shí)。
- 當(dāng)一個(gè)系統(tǒng)不希望使用繼承或因?yàn)槎鄬哟卫^承導(dǎo)致系統(tǒng)類(lèi)的個(gè)數(shù)急劇增加時(shí)。
- 當(dāng)一個(gè)系統(tǒng)需要在構(gòu)件的抽象化角色和具體化角色之間增加更多的靈活性時(shí)。
繼承擁有很多優(yōu)點(diǎn),比如,抽象、封裝、多態(tài)等,父類(lèi)封裝共性,子類(lèi)實(shí)現(xiàn)特性。繼承可以很好的實(shí)現(xiàn)代碼復(fù)用(封裝)的功能,但這也是繼承的一大缺點(diǎn)
由于父類(lèi)擁有的方法,子類(lèi)也會(huì)繼承得到,無(wú)論子類(lèi)需不需要,這說(shuō)明繼承具備強(qiáng)侵入性(父類(lèi)代碼侵入子類(lèi)),同時(shí)會(huì)導(dǎo)致子類(lèi)臃腫,在設(shè)計(jì)模式中,有一個(gè)原則為優(yōu)先使用組合/聚合,而不是繼承。
6.橋接設(shè)計(jì)模式和適配器設(shè)計(jì)模式對(duì)比
橋接模式跟適配器模式,從實(shí)現(xiàn)方式上非常類(lèi)似,橋接設(shè)計(jì)模式是將材質(zhì)這個(gè)一類(lèi)屬性注入到抽象類(lèi)實(shí)現(xiàn)的,適配器設(shè)計(jì)模式是將一個(gè)接口跟目前需求不太匹配的已有組件,通過(guò)適配器的方式進(jìn)行兼容處理的。從這個(gè)角度來(lái)看,都是通過(guò)注入一個(gè)類(lèi)別來(lái)進(jìn)行設(shè)計(jì)的。
但是這兩個(gè)設(shè)計(jì)模式從設(shè)計(jì)的目的上是不同的:橋接設(shè)計(jì)模式是為了分離、解耦合。適配器設(shè)計(jì)模式的目的是為了合并。適配器設(shè)計(jì)模式一般是用在對(duì)之前的已經(jīng)實(shí)現(xiàn)的功能的兼容適配使用。而橋接設(shè)計(jì)模式是從設(shè)計(jì)開(kāi)始的時(shí)候進(jìn)行設(shè)計(jì)的,就已經(jīng)設(shè)計(jì)好了不同的維度。
到此這篇關(guān)于Java設(shè)計(jì)模式中橋接模式應(yīng)用詳解的文章就介紹到這了,更多相關(guān)Java橋接模式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring MVC實(shí)現(xiàn)一次簡(jiǎn)單的CRUD示例
這篇文章主要介紹了Spring MVC實(shí)現(xiàn)一次簡(jiǎn)單的CRUD示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-08-08基于Java的guava開(kāi)源庫(kù)工具類(lèi)
guava是谷歌基于java封裝好的開(kāi)源庫(kù),這篇文章主要通過(guò)介紹幾個(gè)好用的guava工具類(lèi),感興趣的朋友可以參考下面文章內(nèi)容2021-09-09Mybatis中的PageHelper的執(zhí)行流程分析
這篇文章主要介紹了Mybatis的PageHelper執(zhí)行流程,本文給大家介紹介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-02-02Eclipse+Java+Swing+Mysql實(shí)現(xiàn)電影購(gòu)票系統(tǒng)(詳細(xì)代碼)
這篇文章主要介紹了Eclipse+Java+Swing+Mysql實(shí)現(xiàn)電影購(gòu)票系統(tǒng)并附詳細(xì)的代碼詳解,需要的小伙伴可以參考一下2022-01-01基于Jackson實(shí)現(xiàn)API接口數(shù)據(jù)脫敏的示例詳解
用戶的一些敏感數(shù)據(jù),例如手機(jī)號(hào)、郵箱、身份證等信息,在數(shù)據(jù)庫(kù)以明文存儲(chǔ),但在接口返回?cái)?shù)據(jù)給瀏覽器(或三方客戶端)時(shí),希望對(duì)這些敏感數(shù)據(jù)進(jìn)行脫敏,所以本文就給大家介紹以惡如何利用Jackson實(shí)現(xiàn)API接口數(shù)據(jù)脫敏,需要的朋友可以參考下2023-08-08springboot與數(shù)據(jù)庫(kù)返回?cái)?shù)據(jù)中文亂碼
大家好,本篇文章主要講的是springboot與數(shù)據(jù)庫(kù)返回?cái)?shù)據(jù)中文亂碼,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下,方便下次瀏覽2022-01-01Java異步編程之Callbacks與Futures模型詳解
這篇文章主要為大家詳細(xì)介紹了Java異步編程中Callbacks與Futures模型的使用,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-03-03MyBatis學(xué)習(xí)教程(六)-調(diào)用存儲(chǔ)過(guò)程
這篇文章主要介紹了MyBatis學(xué)習(xí)教程(六)-調(diào)用存儲(chǔ)過(guò)程的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,感興趣的朋友一起看下吧2016-05-05