一、概述
裝飾器模式作用是針對(duì)目標(biāo)方法進(jìn)行增強(qiáng),提供新的功能或者額外的功能。
不同于適配器模式和橋接模式,裝飾器模式涉及的是單方,和代理模式相同,而且目標(biāo)必須是抽象的。
而實(shí)際上,裝飾器模式和代理模式的實(shí)現(xiàn)方式基本一致,只在目標(biāo)的存在上有些差別,這個(gè)后面我們具體講述。
二、初步分析
上面提到了兩點(diǎn):
- 涉及的是單方
- 目標(biāo)是抽象的
我們來想一下,所謂單方主要指的是在整個(gè)裝飾器模式中不存在雙方調(diào)用,要解決的也不是雙方調(diào)用的問題,而是解決單方提供對(duì)外服務(wù)的問題,這個(gè)單方在自行對(duì)外提供服務(wù)時(shí),功能不足,或者我們需要額外添加一些新功能,這時(shí)就可以使用裝飾器模式,來對(duì)這個(gè)單方進(jìn)行增強(qiáng)。
目標(biāo)抽象的意思是因?yàn)槲覀冃枰ㄟ^實(shí)現(xiàn)接口的方式來進(jìn)行增強(qiáng),因此目標(biāo)必須抽象為接口。
三、實(shí)例
下面我們用我們生活中的一個(gè)例子來說明,我們用房子來作為目標(biāo):
房子接口:house
1
2
3
4
5
6
|
/** * 目標(biāo)接口:房子 */ public interface house { void output(); } |
具體的房子:donghaohouse
1
2
3
4
5
6
7
8
9
|
/** * 房子實(shí)現(xiàn)類 */ public class donghaohouse implements house { @override public void output() { system.out.println( "這是董浩的房子" ); } } |
具體的房子:donglianghouse
1
2
3
4
5
6
7
8
9
|
/** * 房子實(shí)現(xiàn)類 */ public class donglianghouse implements house { @override public void output() { system.out.println( "這是董量的房子" ); } } |
裝飾器:decorator
1
2
3
4
5
6
7
8
9
10
11
12
|
public class decorator implements house { private house house; public decorator(house house){ this .house = house; } @override public void output() { system.out.println( "這是針對(duì)房子的前段裝飾增強(qiáng)" ); house.output(); system.out.println( "這是針對(duì)房子的后段裝飾增強(qiáng)" ); } } |
測試類:
1
2
3
4
5
6
7
|
public class clienter { public static void main(string[] args) { house donghaohouse = new donghaohouse(); house decorator = new decorator(donghaohouse); decorator.output(); } } |
執(zhí)行結(jié)果為:
這是針對(duì)房子的前段裝飾增強(qiáng)
這是董浩的房子
這是針對(duì)房子的后段裝飾增強(qiáng)
四、解析
通過上面的例子我們可以看出,除了測試類外,只剩下接口和實(shí)現(xiàn)類了,即使是裝飾器類也是目標(biāo)接口的一個(gè)字類,這更能說明單方的說法,模式中所有的類都屬于目標(biāo)方。至于目標(biāo)是抽象的更是如此,只有目標(biāo)是抽象的,才可以使用裝飾器模式來進(jìn)行增強(qiáng)。
上面我們說過裝飾器模式與代理模式基本相同,只存在少許差別。
我們需要從概念上了解代理和裝飾的區(qū)別:
- 代理是全權(quán)代理,目標(biāo)根本不對(duì)外,全部由代理類來完成。
- 裝飾是增強(qiáng),是輔助,目標(biāo)仍然可以自行對(duì)外提供服務(wù),裝飾器只起增強(qiáng)作用。
上面兩點(diǎn)提現(xiàn)到代碼實(shí)現(xiàn)中是這樣的:
代理模式
1
2
3
4
5
6
7
8
9
10
11
12
|
public class proxy implements house { private house house; public decorator(){ this .house = new donghaohouse(); } @override public void output() { system.out.println( "這是針對(duì)目標(biāo)的前段增強(qiáng)" ); house.output(); system.out.println( "這是針對(duì)目標(biāo)的后段增強(qiáng)" ); } } |
裝飾模式
1
2
3
4
5
6
7
8
9
10
11
12
|
public class decorator implements house { private house house; public decorator(house house){ this .house = house; } @override public void output() { system.out.println( "這是針對(duì)房子的前段裝飾增強(qiáng)" ); house.output(); system.out.println( "這是針對(duì)房子的后段裝飾增強(qiáng)" ); } } |
看出來了嗎,裝飾器中持有的目標(biāo)實(shí)例是從構(gòu)造器傳入的,而代理中持有的目標(biāo)實(shí)例是自己創(chuàng)建的。
那么這里又出現(xiàn)一個(gè)區(qū)別,代理模式和裝飾器模式雖然都依賴于目標(biāo)接口,但是代理針對(duì)的目標(biāo)實(shí)現(xiàn)類是固定的,而裝飾器模式可以隨意指定,也就是說目標(biāo)是可以自有擴(kuò)展的。
五、使用場景
裝飾器模式就是使用在對(duì)已有的目標(biāo)功能存在不足,需要增強(qiáng)時(shí),前提是目標(biāo)存在抽象接口。
六、總結(jié)
我們要明白代理模式和裝飾器模式的區(qū)別,區(qū)分二者的使用場景,如下圖:
以上所述是小編給大家介紹的java設(shè)計(jì)模式系列-裝飾器模式詳解整合,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)服務(wù)器之家網(wǎng)站的支持!
原文鏈接:https://www.jianshu.com/p/45fc8a521a44