裝飾者模式在不必改變?cè)愇募褪褂美^承的情況下,動(dòng)態(tài)地?cái)U(kuò)展一個(gè)對(duì)象的功能。它是通過創(chuàng)建一個(gè)包裝對(duì)象,也就是裝飾來包裹真實(shí)的對(duì)象。 裝飾者模式特點(diǎn)裝飾者和被裝飾對(duì)象有相同的超類型。 你可以用一個(gè)或者多個(gè)裝飾者包裝一個(gè)對(duì)象。 既然裝飾者和被裝飾對(duì)象有相同的超類型,所以在任何需要原始對(duì)象(被包裝的)的場(chǎng)合,可以用裝飾過的對(duì)象代替它。 裝飾者可以在所委托被裝飾者的行為之前與/或之后,加上自己的行為,以達(dá)到特定的目的。 對(duì)象可以在任何時(shí)候被裝飾,所以可以在運(yùn)行時(shí)動(dòng)態(tài)的,不限量的用你喜歡的裝飾者來裝飾對(duì)象。
裝飾者模式類圖
從上圖可以看出,裝飾者和被裝飾者擁有共同的父類Componet,為了方便后面的擴(kuò)展,我們加了一個(gè)裝飾者父類Decorator,讓Decorator去繼承Componet,然后讓眾多裝飾者去繼承Decorator。 應(yīng)用場(chǎng)景(婚紗照收費(fèi))生活中符合裝飾者模式的應(yīng)用場(chǎng)景其實(shí)還挺多,就拿前陣子去拍婚紗照的經(jīng)歷來說吧,婚紗工作人員一般先會(huì)跟你整體說,拍什么樣的場(chǎng)景,多少套衣服,怎么去收費(fèi),然后在不改變?cè)惺召M(fèi)的前提下,會(huì)跟你說精修入冊(cè)數(shù)量,每加一張,就加收多少錢?,F(xiàn)在我們來把這個(gè)分級(jí)收費(fèi)模式抽象一下,假如說,我們拍攝一張照片是40塊錢,精修一張40塊,入冊(cè)10塊。安插在裝飾者模式中,照片本身就成了我們說的被裝飾者(ConcreteComponet),而精修和入冊(cè)則符合我們對(duì)裝飾者的定義,分別為DecoratorA,DecoratorB。接下來我們來實(shí)現(xiàn)一下這個(gè)應(yīng)用場(chǎng)景的類圖結(jié)構(gòu)。 | 照片本身 | 照片入冊(cè) | 照片精修 | 費(fèi)用 | 40 | 10 | 40 |
場(chǎng)景(婚紗照收費(fèi))類圖
Format為裝飾者父類與WeddingPhoto(被裝飾者)共同繼承了Photo。IsPhotoAlbum(入冊(cè))與PhotoWithPs(精修)為裝飾者,共同修飾WeddingPhoto。他們都擁有共同的行為cost(),用于計(jì)算出價(jià)格,description屬性主要用于記錄信息,方便查看實(shí)現(xiàn)結(jié)果,可要可不要。 代碼實(shí)現(xiàn)共同抽象父類package decorate.base;
/**
* 照片父類
* @author vision
*/
public abstract class Photo {
private String description="";
public String getDescription() {
return description;
}
public void setDescription(String description) {
this.description = description;
}
/**
* 價(jià)格
* @param price
* @return
*/
public abstract Double cost();
} 裝飾者抽象父類package decorate.format;
import decorate.base.Photo;
/**
* Format(照片規(guī)格)裝飾者,擴(kuò)展自Photo父類
* @author vision
*/
public abstract class Format extends Photo{
/**
* 描述
* @param description
* @return
*/
public abstract String getDescription();
} 被裝飾者(婚紗照)package decorate;
import decorate.base.Photo;
/**
* 被裝飾者:婚紗照
* @author vision
*/
public class WeddingPhoto extends Photo{
public WeddingPhoto() {
setDescription(getDescription()+"WeddingPhoto(婚紗照)");
}
@Override
public Double cost() {
return 40.0;
}
} 裝飾者(入冊(cè))package decorate.format;
import decorate.base.Photo;
/**
* 裝飾者:照片是否入冊(cè)
* @author vision
*/
public class IsPhotoAlbum extends Format{
private Photo photo;
public IsPhotoAlbum(Photo photo) {
this.photo=photo;
}
@Override
public String getDescription() {
return photo.getDescription()+"+IsPhotoAlbum(入冊(cè))";
}
@Override
public Double cost() {
// TODO Auto-generated method stub
return 10.0+photo.cost();
}
} 裝飾者(精修)package decorate.format;
import decorate.base.Photo;
/**
* 裝飾者:照片精修
* @author vision
*/
public class PhotoWithPs extends Format{
private Photo photo;
public PhotoWithPs(Photo photo){
this.photo=photo;
}
@Override
public String getDescription() {
return photo.getDescription()+"+PhotoWithPs(精修)";
}
/**
* 照片精修,每張多加40塊
*/
@Override
public Double cost() {
return 40.0+photo.cost();
}
} 測(cè)試類package decorate;
import decorate.base.Photo;
import decorate.format.IsPhotoAlbum;
import decorate.format.PhotoWithPs;
public class TestClass {
public static void main(String[] args) {
Photo weddingPhoto=new WeddingPhoto();
//精修
PhotoWithPs photoWithPs=new PhotoWithPs(weddingPhoto);
System.out.println("最終選擇的照片為:"+photoWithPs.getDescription());
System.out.println("最終單價(jià)為:"+photoWithPs.cost());
//入冊(cè)
IsPhotoAlbum isPhotoAlbum=new IsPhotoAlbum(photoWithPs);
System.out.println("最終選擇的照片為:"+isPhotoAlbum.getDescription());
System.out.println("最終單價(jià)為:"+isPhotoAlbum.cost());
}
} 輸出結(jié)果
備注參考資料:《Head First 設(shè)計(jì)模式》
|