Java裝飾者模式的深入了解
一、裝飾模式的定義和特點(diǎn)
在軟件開發(fā)過程中,有時(shí)想用一些現(xiàn)存的組件。這些組件可能只是完成了一些核心功能。但在不改變其結(jié)構(gòu)的情況下,可以動(dòng)態(tài)地?cái)U(kuò)展其功能。所有這些都可以釆用裝飾器模式來實(shí)現(xiàn)。
就像我們做菜,需要用到調(diào)料,菜,刀,火等一系列抽象的組件來最終完成一道菜。
裝飾模式的定義:
指在不改變現(xiàn)有對(duì)象結(jié)構(gòu)的情況下,動(dòng)態(tài)地給該對(duì)象增加一些職責(zé)(即增加其額外功能)的模式,它屬于對(duì)象結(jié)構(gòu)型模式。就增加功能來說,裝飾模式比生成子類更加靈活。
特點(diǎn):
裝飾器是繼承的有力補(bǔ)充,比繼承靈活,在不改變?cè)袑?duì)象的情況下,動(dòng)態(tài)的給一個(gè)對(duì)象擴(kuò)展功能,即插即用
通過使用不用裝飾類及這些裝飾類的排列組合,可以實(shí)現(xiàn)不同效果
裝飾器模式完全遵守開閉原則
缺點(diǎn)
裝飾器模式會(huì)增加許多子類,過度使用會(huì)增加程序得復(fù)雜性。
二、裝飾模式的結(jié)構(gòu)
裝飾模式的結(jié)構(gòu)一般包含以下幾個(gè)角色
1. 抽象構(gòu)件(Component)角色:定義一個(gè)抽象接口以規(guī)范準(zhǔn)備接收附加責(zé)任的對(duì)象。
2. 具體構(gòu)件(ConcreteComponent)角色:實(shí)現(xiàn)抽象構(gòu)件,通過裝飾角色為其添加一些職責(zé)。
3. 抽象裝飾(Decorator)角色:繼承抽象構(gòu)件,并包含具體構(gòu)件的實(shí)例,可以通過其子類擴(kuò)展具體構(gòu)件的功能。
4. 具體裝飾(ConcreteDecorator)角色:實(shí)現(xiàn)抽象裝飾的相關(guān)方法,并給具體構(gòu)件對(duì)象添加附加的責(zé)任。
圖示

三、咖啡點(diǎn)單案例演示
有一個(gè)需求,點(diǎn)一杯咖啡需要咖啡,材料等等,這個(gè)案例就很適合裝飾模式,類似于穿衣,點(diǎn)餐,買包子,等等,我們?cè)趺窗阉O(shè)計(jì)成裝飾模式呢?
看類圖

這個(gè)結(jié)構(gòu)就是我已經(jīng)設(shè)計(jì)好的一個(gè)裝飾模式的類圖,idea自動(dòng)生成的,這里的Drink就是我們上面說的抽象構(gòu)建角色,裝飾者是Decorator,他是一個(gè)抽象裝飾,下面他的子類就是具體的裝飾者,那么具體構(gòu)建中間我們提供了一個(gè)中間構(gòu)建,提供了coffee的一些共性,可以放在這里,用的時(shí)候直接繼承,他的下面就是相應(yīng)的具體構(gòu)件,具體被裝飾者角色,裝飾者與被裝飾者共同繼承自component抽象構(gòu)件,需要用到裝飾的就是我們點(diǎn)一杯咖啡,用裝飾去包裹即可,層層包裹,案例如下:
比如我要點(diǎn)一份加糖加奶的拿鐵咖啡
代碼實(shí)例:
component抽象構(gòu)件角色:
package com.decoratorPattern.starBucks;
/**
* @author wang
* @version 1.0
* @packageName com.decoratorPattern.starBucks
* @className Drink
* @date 2021/12/28 10:28
* @Description 飲料構(gòu)件類抽象component
*/
public abstract class Drink {
private String description;
private float price = 0.0f;
public String getDescription() {
return description;
}
public void setDescription(String description) {
this.description = description;
}
public float getPrice() {
return price;
}
public void setPrice(float price) {
this.price = price;
}
/**
* @Date 2021/12/28 10:30
* @Param
* @Return float
* @MetodName cost
* @Author wang
* @Description 計(jì)算花費(fèi),訂單價(jià)格
*/
public abstract float cost();
}裝飾者類:
package com.decoratorPattern.starBucks;
/**
* @author wang
* @version 1.0
* @packageName com.decoratorPattern.starBucks
* @className Decorator
* @date 2021/12/28 10:40
* @Description 裝飾者定義類,配料
*/
public class Decorator extends Drink {
private Drink drink;
/**
* @param drink
* @Date 2021/12/28 10:42
* @Param
* @Return null
* @MetodName Decorator
* @Author wang
* @Description 傳入一個(gè)被裝飾者,由裝飾者進(jìn)行裝飾
*/
public Decorator(Drink drink) {
this.drink = drink;
}
/**
* @Date 2021/12/28 10:43
* @Param
* @Return float
* @MetodName cost
* @Author wang
* @Description 裝飾者的價(jià)格加上被裝飾者的價(jià)格
*/
@Override
public float cost() {
return super.getPrice() + drink.cost();
}
/**
* @Date 2021/12/28 10:44
* @Param
* @Return String
* @MetodName getDescription
* @Author wang
* @Description 輸出訂單信息,包含裝飾者,裝飾者的價(jià)格,以及被裝飾者的信息
*/
@Override
public String getDescription() {
return drink.getDescription() + "\n加入的材料:" + super.getDescription()
+ "\t材料價(jià)格:" + super.getPrice() ;
}
}裝飾者類:
package com.decoratorPattern.starBucks;
/**
* @author wang
* @version 1.0
* @packageName com.decoratorPattern.starBucks
* @className Decorator
* @date 2021/12/28 10:40
* @Description 裝飾者定義類,配料
*/
public class Decorator extends Drink {
private Drink drink;
/**
* @param drink
* @Date 2021/12/28 10:42
* @Param
* @Return null
* @MetodName Decorator
* @Author wang
* @Description 傳入一個(gè)被裝飾者,由裝飾者進(jìn)行裝飾
*/
public Decorator(Drink drink) {
this.drink = drink;
}
/**
* @Date 2021/12/28 10:43
* @Param
* @Return float
* @MetodName cost
* @Author wang
* @Description 裝飾者的價(jià)格加上被裝飾者的價(jià)格
*/
@Override
public float cost() {
return super.getPrice() + drink.cost();
}
/**
* @Date 2021/12/28 10:44
* @Param
* @Return String
* @MetodName getDescription
* @Author wang
* @Description 輸出訂單信息,包含裝飾者,裝飾者的價(jià)格,以及被裝飾者的信息
*/
@Override
public String getDescription() {
return drink.getDescription() + "\n加入的材料:" + super.getDescription()
+ "\t材料價(jià)格:" + super.getPrice() ;
}
}具體構(gòu)件類:拿鐵
package com.decoratorPattern.starBucks;
/**
* @author wang
* @version 1.0
* @packageName com.decoratorPattern.starBucks
* @className latte
* @date 2021/12/28 10:32
* @Description 拿鐵咖啡實(shí)類,被裝飾者
*/
public class Latte extends Coffee{
public Latte() {
setDescription("拿鐵咖啡");
setPrice(15.0f);
}
}具體構(gòu)件類:摩卡
package com.decoratorPattern.starBucks;
/**
* @author wang
* @version 1.0
* @packageName com.decoratorPattern.starBucks
* @className Mocha
* @date 2021/12/28 10:36
* @Description 摩卡咖啡實(shí)類,被裝飾者
*/
public class Mocha extends Coffee {
public Mocha() {
setDescription("摩卡咖啡");
setPrice(12.2f);
}
}其他同上,不過多展示
具體裝飾類:牛奶
package com.decoratorPattern.starBucks;
/**
* @author wang
* @version 1.0
* @packageName com.decoratorPattern.starBucks
* @className Milk
* @date 2021/12/28 10:47
* @Description 牛奶調(diào)味品,具體裝飾者
*/
public class Milk extends Decorator{
/**
* @param drink
* @Date 2021/12/28 10:42
* @Param
* @Return null
* @MetodName Decorator
* @Author wang
* @Description 傳入一個(gè)被裝飾者,由裝飾者進(jìn)行裝飾
*/
public Milk(Drink drink) {
super(drink);
setDescription("牛奶");
setPrice(1.0f);
}
}具體裝飾:糖
package com.decoratorPattern.starBucks;
/**
* @author wang
* @version 1.0
* @packageName com.decoratorPattern.starBucks
* @className sugar
* @date 2021/12/28 10:50
* @Description 糖,裝飾者
*/
public class Sugar extends Decorator{
/**
* @param drink
* @Date 2021/12/28 10:42
* @Param
* @Return null
* @MetodName Decorator
* @Author wang
* @Description 傳入一個(gè)被裝飾者,由裝飾者進(jìn)行裝飾
*/
public Sugar(Drink drink) {
super(drink);
setDescription("糖");
setPrice(0.5f);
}
}訂單測(cè)試代碼:
package com.decoratorPattern.starBucks;
/**
* @author wang
* @version 1.0
* @packageName com.decoratorPattern.starBucks
* @className OrderTest
* @date 2021/12/28 10:51
* @Description 前臺(tái)訂單類
*/
public class OrderTest {
public static void main(String[] args) {
//點(diǎn)一份加糖加奶的拿鐵咖啡
System.out.println("+++++++沒加任何東西+++++++");
Drink latte = new Latte();
System.out.println("當(dāng)前總價(jià):" + latte.cost());
System.out.println("coffee:" +latte.getDescription());
//加糖
System.out.println("+++++++加糖后+++++++");
latte = new Sugar(latte);
System.out.println("當(dāng)前總價(jià):" + latte.cost());
System.out.println("coffee:" + latte.getDescription());
System.out.println("+++++++加奶后+++++++");
latte = new Milk(latte);
System.out.println("當(dāng)前總價(jià):" + latte.cost());
System.out.println("coffee:" +latte.getDescription());
}
}
/**
* +++++++沒加任何東西+++++++
* 當(dāng)前總價(jià):15.0
* coffee:拿鐵咖啡
* +++++++加糖后+++++++
* 當(dāng)前總價(jià):15.5
* coffee:拿鐵咖啡
* 加入的材料:糖 材料價(jià)格:0.5
* +++++++加奶后+++++++
* 當(dāng)前總價(jià):16.5
* coffee:拿鐵咖啡
* 加入的材料:糖 材料價(jià)格:0.5
* 加入的材料:牛奶 材料價(jià)格:1.0
*
* Process finished with exit code 0
*/綜上, 如果我們需要新的咖啡種類或者是新的調(diào)料,只需要新增類去繼承coffee或者decorator類即可。
四、總結(jié)
裝飾模式是為已有的功能動(dòng)態(tài)的添加更多功能的一種方式,當(dāng)系統(tǒng)需要新功能的時(shí)候,向舊的類中添加新的代碼,這些新加的代碼通常裝飾了原有類的核心職責(zé)或主要行為。
優(yōu)點(diǎn):
把類中裝飾功能從類中移除,這樣可以簡化原來的類,
有效的把類的核心職責(zé)和裝飾功能分開了,而且可以去除相關(guān)類中的重復(fù)裝飾邏輯
可代替繼承。
總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
詳解SpringBoot啟動(dòng)類的掃描注解的用法及沖突原則
這篇文章主要介紹了詳解SpringBoot啟動(dòng)類的掃描注解的用法及沖突原則,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11
Servlet會(huì)話技術(shù)基礎(chǔ)解析
這篇文章主要介紹了Servlet會(huì)話技術(shù)基礎(chǔ)解析,具有一定借鑒價(jià)值,需要的朋友可以參考下。2017-12-12
spring AOP實(shí)現(xiàn)@Around輸出請(qǐng)求參數(shù)和返回參數(shù)
這篇文章主要介紹了spring AOP實(shí)現(xiàn)@Around輸出請(qǐng)求參數(shù)和返回參數(shù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02
Java動(dòng)態(tài)規(guī)劃篇之線性DP的示例詳解
這篇文章主要通過幾個(gè)例題為大家詳細(xì)介紹一些Java動(dòng)態(tài)規(guī)劃中的線性DP,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Java有一定的幫助,需要的可以參考一下2022-11-11
利用Java實(shí)現(xiàn)和可被K整除的子數(shù)組完整實(shí)例
這篇文章主要給大家介紹了關(guān)于利用Java實(shí)現(xiàn)和可被K整除的子數(shù)組的相關(guān)資料,這道題來自力扣,通過學(xué)習(xí)這道題的解題思路以及代碼對(duì)大家的學(xué)習(xí)或者工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2024-01-01

