欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

.Net行為型設(shè)計模式之職責(zé)鏈模式(Chain of Responsibility)

 更新時間:2022年05月26日 11:11:17   作者:springsnow  
這篇文章介紹了.Net行為型設(shè)計模式之職責(zé)鏈模式(Chain of Responsibility),文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下

一、動機(Motivate)

在軟件構(gòu)建過程中,一個請求可能被多個對象處理,但是每個請求在運行時只能有一個接受者,如果顯示指定,將必不可少地帶來請求發(fā)送者與接受者的緊耦合。如何使請求的發(fā)送者不需要指定具體的接受者,讓請求的接受者自己在運行時決定來處理請求,從而使兩者解耦。

二、意圖(Intent)

避免請求發(fā)送者與接收者耦合在一起,讓多個對象都有可能接受請求,將這些對象連接成一條鏈,并且沿著這條鏈傳遞請求,知道有對象處理它為止。                                 ——《設(shè)計模式》GoF

三、結(jié)構(gòu)圖(Structure)

四、模式的組成

可以看出,在職責(zé)鏈模式的結(jié)構(gòu)圖有以下角色:
(1)、抽象處理者角色(Handler):抽象處理者定義了一個處理請求的接口,它一般設(shè)計為抽象類,由于不同的具體處理者處理請求的方式不同,因此在其中定義了抽象請求處理方法。因為每一個處理者的下家還是一個處理者,因此在抽象處理者中定義了一個自類型的對象,作為其對下家的引用。通過該引用,處理者可以連成一條鏈。
(2)、具體處理者角色(ConcreteHandler):具體處理者是抽象處理者的子類,它可以處理用戶請求,在具體處理者類中實現(xiàn)了抽象處理者中定義的抽象處理方法,在處理請求之前需要進(jìn)行判斷,看是否有相應(yīng)的處理權(quán)限,如果可以處理請求就處理它,否則將請求轉(zhuǎn)發(fā)給后繼者;在具體處理者中可以訪問鏈中下一個對象,以便請求的轉(zhuǎn)發(fā)。

五、職責(zé)鏈模式的代碼實現(xiàn)

在現(xiàn)實生活中,職責(zé)鏈模式的例子也是很多的,例如:公司的請假流程就是一個很好的職責(zé)鏈模式的例子,如果請假半天,只要告訴本部門經(jīng)理就可以了;如果請假7天或者以上必須人事總監(jiān)批準(zhǔn);如果請假15天以上,那就要經(jīng)過總裁批準(zhǔn)了。還有類似的例子就是采購的流程,其流程也是職責(zé)鏈模式很好的體現(xiàn),采購的金額不同,需要批準(zhǔn)的人員也不同,比如:部門采購1萬元的紙品,只要部門領(lǐng)導(dǎo)簽批就可以,如果要采購大于1萬小于5萬的物品,那就需要財務(wù)經(jīng)理簽批了,如果采購30萬的原材料或者物品,那就需要總裁或者類似角色才能審批了。接下來我們就以采購的實例來說明職責(zé)鏈模式。實現(xiàn)代碼如下:

static void Main(string[] args)
{
    PurchaseRequest requestDao = new PurchaseRequest(8000.0, "單刀5把");
    PurchaseRequest requestHuaJi = new PurchaseRequest(10000.0, "10把方天畫戟");
    PurchaseRequest requestJian = new PurchaseRequest(80000.0, "5把金絲龍鱗閃電劈");

    Approver manager = new Manager("黃飛鴻");
    Approver financial = new FinancialManager("黃麒英");
    Approver ceo = new CEO("十三姨");

    // 設(shè)置職責(zé)鏈
    manager.NextApprover = financial;
    financial.NextApprover = ceo;

    // 處理請求
    manager.ProcessRequest(requestDao);
    manager.ProcessRequest(requestHuaJi);
    manager.ProcessRequest(requestJian);

}
// 采購請求
public sealed class PurchaseRequest
{
    // 金額
    public double Amount { get; set; }

    // 產(chǎn)品名字
    public string ProductName { get; set; }

    public PurchaseRequest(double amount, string productName)
    {
        Amount = amount;
        ProductName = productName;
    }
}

//抽象審批人,Handler---相當(dāng)于“抽象處理者角色”
public abstract class Approver
{
    //下一位審批人,由此形成一條鏈
    public Approver NextApprover { get; set; }

    //審批人的名稱
    public string Name { get; set; }

    public Approver(string name)
    {
        this.Name = name;
    }

    //處理請求
    public abstract void ProcessRequest(PurchaseRequest request);
}

//部門經(jīng)理----相當(dāng)于“具體處理者角色” ConcreteHandler
public sealed class Manager : Approver
{
    public Manager(string name) : base(name) { }

    public override void ProcessRequest(PurchaseRequest request)
    {
        if (request.Amount <= 10000.0)
        {
            Console.WriteLine("{0} 部門經(jīng)理批準(zhǔn)了對原材料{1}的采購計劃!", this.Name, request.ProductName);
        }
        else if (NextApprover != null)
        {
            NextApprover.ProcessRequest(request);
        }
    }
}

//財務(wù)經(jīng)理---相當(dāng)于“具體處理者角色”ConcreteHandler
public sealed class FinancialManager : Approver
{
    public FinancialManager(string name) : base(name) { }

    public override void ProcessRequest(PurchaseRequest request)
    {
        if (request.Amount > 10000.0 && request.Amount <= 50000.0)
        {
            Console.WriteLine("{0} 財務(wù)經(jīng)理批準(zhǔn)了對原材料{1}的采購計劃!", this.Name, request.ProductName);
        }
        else if (NextApprover != null)
        {
            NextApprover.ProcessRequest(request);
        }
    }
}

//總裁---相當(dāng)于“具體處理者角色” ConcreteHandler
public sealed class CEO : Approver
{
    public CEO(string name) : base(name) { }

    public override void ProcessRequest(PurchaseRequest request)
    {
        if (request.Amount > 50000.0 && request.Amount < 300000.0)
        {
            Console.WriteLine("{0} 總裁批準(zhǔn)了對原材料 {1} 的采購計劃!", this.Name, request.ProductName);
        }
        else
        {
            Console.WriteLine("這個采購計劃的金額比較大,需要一次董事會會議討論才能決定!");
        }
    }
}

六、職責(zé)鏈模式的實現(xiàn)要點:

Chain of Responsibility模式的應(yīng)用場合在于“一個請求可能有多個接受者,但是最后真正的接受者只有一個”,只有這時候請求發(fā)送者與接受者的耦合才有可能出現(xiàn)“變化脆弱”的癥狀,職責(zé)鏈的目的就是將二者解耦,從而更好地應(yīng)對變化。
應(yīng)用了Chain of Responsibility模式后,對象的職責(zé)分派將更具靈活性。我們可以在運行時動態(tài)添加/修改請求的處理職責(zé)。
當(dāng)我們要新增一個DHandler處理請求,就不需再改原來的代碼了,遵從了開放封閉原則。這樣我們的程序就更賦予變化,更有變化的抵抗力。Handler類本身繼承自BaseHandler類型,又包含了一個BaseHandler類型的對象,這點類似Decorator模式。
如果請求傳遞到職責(zé)鏈的末尾仍得不到處理,應(yīng)該有一個合理的缺省機制。這也是每一個接受對象的責(zé)任,而不是發(fā)出請求的對象的責(zé)任。

1、職責(zé)鏈模式的主要優(yōu)點有:

1】、降低耦合度:職責(zé)鏈模式使得一個對象無需知道是其他哪一個對象處理其請求。對象僅需知道該請求會被處理即可,接受者和發(fā)送者都沒有對方的明確信息,且鏈中的對象不需要知道鏈的結(jié)構(gòu),有客戶端負(fù)責(zé)鏈的創(chuàng)建。
2】、可簡化對象的相互連接:接受者對象僅需維持一個指向其后繼者的引用,而不需維持它對所有的候選處理者的引用。
3】、增強給對象指派職責(zé)的靈活性:在給對象分派職責(zé)時,職責(zé)鏈可以給我們帶來更多的靈活性??梢酝ㄟ^在運行時對該連進(jìn)行動態(tài)的增加或修改處理一個請求的職責(zé)。
4】、增加新的請求處理類很方便:在系統(tǒng)中增加一個新的請求處理者無需修改原有系統(tǒng)的代碼,只需要在客戶端重新建鏈即可,從這一點看來是符合“開閉原則”的。

2、職責(zé)鏈模式的主要缺點有:

1】、在找到正確的處理對象之前,所有的條件判定都要執(zhí)行一遍,當(dāng)責(zé)任鏈過長時,可能會引起性能的問題。
2】、可能導(dǎo)致某個請求不被處理。
3】、客戶端需要組裝這個鏈條,耦合了客戶端和鏈條的組成結(jié)構(gòu),可以把這個在客戶端的組合動作提到外面,通過配置來做,會更好點。

3、在下面的情況下可以考慮使用職責(zé)鏈模式:

1】、一個系統(tǒng)的審批需要多個對象才能完成處理的情況下,例如請假系統(tǒng)等。
2】、代碼中存在多個if-else語句的情況下,此時可以考慮使用責(zé)任鏈模式來對代碼進(jìn)行重構(gòu)
3】、有多個對象可以處理同一個請求,具體哪個對象處理該請求有運行時刻自動確定。客戶端只需將請求提交到鏈上,無須關(guān)心請求的處理對象是誰以及它是如何處理的。
4】、不明確指定接受者的情況下,向多個對象中的一個提交一個請求。請求的發(fā)送者與請求者解耦,請求將沿著鏈進(jìn)行傳遞,尋求響應(yīng)的處理者。
5】、可動態(tài)指定一組對象處理請求。客戶端可以動態(tài)創(chuàng)建職責(zé)鏈來處理請求,還可以動態(tài)改變鏈中處理者之間的先后次序

七、.NET 職責(zé)鏈模式的實現(xiàn)

這個模式在Net框架中的實現(xiàn)不多,我感覺這個模式的使用場景更多的是在業(yè)務(wù)系統(tǒng)總才會有更大的用處。這種模式在處理UI的消息時很常用,但實際上Windows消息循環(huán)還是硬編碼的結(jié)構(gòu)。因為效率上的考慮,Windows消息循環(huán)是哪個對象有一個請求,則直接到達(dá)處理函數(shù)的地址。如果鏈條上的對象多了,而真正處理的函數(shù)在鏈條后部分,效率會很低下。因此我們在使用這種模式的時候更適合業(yè)務(wù)流程,即對性能要求不是特別高的情況更加常用。

到此這篇關(guān)于.Net行為型設(shè)計模式之職責(zé)鏈模式(Chain of Responsibility)的文章就介紹到這了。希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • asp.net Ajax之無刷新評論介紹

    asp.net Ajax之無刷新評論介紹

    asp.net Ajax之無刷新評論介紹;需要的朋友可以參考下
    2012-11-11
  • ASP.Net生成一個簡單的圖片

    ASP.Net生成一個簡單的圖片

    ASP.Net生成一個簡單的圖片...
    2006-10-10
  • ASP.NET中使用用戶控件

    ASP.NET中使用用戶控件

    這篇文章介紹了ASP.NET中使用用戶控件的方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-05-05
  • ASP.NET Session使用詳解

    ASP.NET Session使用詳解

    Session是什么呢?簡單來說就是服務(wù)器給客戶端的一個編號。當(dāng)一臺WWW服務(wù)器運行時,可能有若干個用戶瀏覽正在運正在這臺服務(wù)器上的網(wǎng)站。
    2009-02-02
  • ASP.NET中Literal控件的使用方法

    ASP.NET中Literal控件的使用方法

    本文主要介紹Literal的使用方法,并簡單列出和Label控件之間的區(qū)別,希望能幫助大家選擇使用。
    2016-04-04
  • ASP.NET?Core的中間件與管道介紹

    ASP.NET?Core的中間件與管道介紹

    這篇文章介紹了ASP.NET?Core的中間件與管道,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-01-01
  • ASP.Net?Core?MVC基礎(chǔ)系列之環(huán)境設(shè)置

    ASP.Net?Core?MVC基礎(chǔ)系列之環(huán)境設(shè)置

    這篇文章介紹了ASP.Net?Core?MVC環(huán)境設(shè)置的方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-02-02
  • ASP.NET2.0+SQL Server2005構(gòu)建多層應(yīng)用

    ASP.NET2.0+SQL Server2005構(gòu)建多層應(yīng)用

    ASP.NET2.0+SQL Server2005構(gòu)建多層應(yīng)用...
    2006-12-12
  • ASP.NET?Core?MVC中的布局(Layout)

    ASP.NET?Core?MVC中的布局(Layout)

    這篇文章介紹了ASP.NET?Core?MVC中的布局(Layout),文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-04-04
  • asp.net中MVC的處理流程詳解

    asp.net中MVC的處理流程詳解

    本文詳細(xì)講解了asp.net MVC的處理流程,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07

最新評論