前言
可能對設(shè)計(jì)模式學(xué)習(xí)尚淺行您,所以每次回味“Decorator”模式時(shí)都覺的這是一個(gè)天才的設(shè)計(jì)模式(可能僅對于我自己來講),所以首先我們先探討一下使用“Decorator”模式的動(dòng)機(jī):當(dāng)我們過度地使用繼承來擴(kuò)展對象的功能時(shí)岔帽,由于繼承為類型引入的靜態(tài)特質(zhì)赎懦,使得這種擴(kuò)展方式缺乏靈活性,并且隨著子類的增多(擴(kuò)展功能的增多)檬洞,各種子類的組合會(huì)導(dǎo)致更多子類的膨脹,因此為了避免這種膨脹我們引入了“Decorator”(裝飾)模式雨效。
使用手法:裝飾類和被裝飾類都繼承同一接口迅涮,被裝飾類的對象傳入裝飾類的構(gòu)造函數(shù)中當(dāng)作參數(shù),在執(zhí)行被裝飾業(yè)務(wù)之前加入裝飾類業(yè)務(wù)邏輯即可徽龟!
1.“Decorator”模式定義:動(dòng)態(tài)地給一個(gè)對象增加一些額外地職責(zé)叮姑。就增加功能而言,Decorator模式比生成子類更靈活据悔。
2.“Decorator”模式要點(diǎn):
1.)通過采用組合传透,而非繼承地手法,Decorator模式實(shí)現(xiàn)了在運(yùn)動(dòng)時(shí)動(dòng)態(tài)地?cái)U(kuò)展對象功能地能力极颓,而且可以根據(jù)需要擴(kuò)展多個(gè)功能朱盐。避免了單獨(dú)使用繼承帶來地“靈活性差”和“多子類衍生問題”。
2.)上圖中的Component類在Decorator模式中充當(dāng)抽象接口的角色菠隆,不應(yīng)該去實(shí)現(xiàn)具體的行為兵琳。而且Decorator類對于Component類應(yīng)該透明,換言之Component類無需知道Decorator類浸赫,Decorator類是從外部來擴(kuò)展Component類的功能闰围。
3.)Decorator類在接口上表現(xiàn)為is-a Component的繼承關(guān)系,即Decorator類繼承了Component類所具有的接口既峡。但在實(shí)現(xiàn)上又表現(xiàn)為has-a Component組合關(guān)系,即Decorator類又使用了另一個(gè)Component類碧查。我們可以使用一個(gè)或多個(gè)Decorator對象來“裝飾”一個(gè)Component對象运敢,且裝飾后的對象仍然是一個(gè)Component對象。
4.)Decorator模式并非解決“多子類衍生的多繼承”問題忠售,Decorator模式應(yīng)用的要點(diǎn)在于解決“主體在多個(gè)方向上的擴(kuò)展功能”传惠。
下面我們看一個(gè)如何使用裝飾給坦克擴(kuò)展“衛(wèi)星定位功能”和“降噪功能”:
//抽象類,相當(dāng)于上圖中的Componen
public abstract class Tank
{
public abstract void Shot();
public abstract void Run();
}
//實(shí)現(xiàn)類稻扬,相當(dāng)于上圖的“ConcreteComponent”
public class T50 : Tank
{
public override void Run()
{
throw new NotImplementedException();
}
public override void Shot()
{
throw new NotImplementedException();
}
}
//Decorator裝飾
public abstract class Decorator:Tank
{
Tank tank;
public Decorator(Tank tank)
{
this.tank = tank;
}
public override void Shot()
{
throw new NotImplementedException();
}
public override void Run()
{
throw new NotImplementedException();
}
}
/// <summary>
/// 添加衛(wèi)星定位功能
/// </summary>
public class DecoratorA : Decorator
{
public DecoratorA(Tank tank) : base(tank)
{
}
public override void Run()
{
//衛(wèi)星定位邏輯代碼
//...
base.Run();
}
public override void Shot()
{
//衛(wèi)星定位邏輯代碼
//...
base.Shot();
}
}
/// <summary>
/// 添加降噪功能
/// </summary>
public class DecoratorB: Decorator
{
public DecoratorB(Tank tank) : base(tank)
{
}
public override void Run()
{
//降噪功能邏輯代碼
//...
base.Run();
}
public override void Shot()
{
//降噪功能邏輯代碼
//...
base.Shot();
}
}
//客戶程序調(diào)用
public class App
{
public static void Main(string[] args)
{
T50 tank = new T50();
//T50實(shí)現(xiàn)衛(wèi)星定位功能
DecoratorA da = new DecoratorA(tank);
//T50實(shí)現(xiàn)降噪功能
DecoratorB db = new DecoratorB(da);
}
}
在.Net框架中“Decorator”模式也經(jīng)常使用卦方,例如:
Stream下“BufferedStream”和“CryptoStream”為裝飾對象,而“FileStream”泰佳、“NetworkStream”盼砍、“MemoryStream”為具體實(shí)現(xiàn)類。