iOS - 責任鏈模式

定義

責任鏈模式:使多個對象都有機會處理請求胜嗓,從而避免了請求的發(fā)送者和接受者之間的耦合關系伏社。將這些對象連成一條鏈短条,并沿著這條鏈傳遞該請求岩四,直到有對象處理它為止恭朗。

責任鏈模式的重點是在“鏈”上屏镊,有一條鏈去處理相似的請求,在鏈中決定誰來處理這個請求痰腮,并返回相應的結(jié)果而芥。

類圖

607542-20151027145216575-700989924.png
  • Handle是上層抽象類,定義handleRequest方法(處理他知道如何讓處理的請求對象)膀值。Handle有一個指向另一個同類型實例的引用棍丐,即successor。當調(diào)用Handler實例的handleRequest方法不知道如何處理請求時沧踏,它會用同樣的方法把請求轉(zhuǎn)發(fā)給successor歌逢,一直如此,直到傳給鏈中的最后一個Handler

  • ConcreteHandler1和ConcreteHandler2實現(xiàn)了handleRequest方法(處理他們認識的請求對象)翘狱。

看一個形象點的圖

2167733-5fbbc1275233d9fa.png

就好比如上圖:由發(fā)送者發(fā)起請求秘案,由具體的接受者處理該請求,先是一級接受者進行處理潦匈,如果它能夠處理就直接處理請求阱高,如果不能處理就傳遞到二級接受者,二級接受者同樣看是否自己能夠處理該請求茬缩,如果能夠處理就執(zhí)行不能處理就傳遞給下一級赤惊。

跟我們平時開發(fā)找誰解決bug是一樣的,比如:團隊中有3個 iOS開發(fā)人員(ABC) 寒屯,測試人員發(fā)現(xiàn)了一個 bug 荐捻,然后先問A:“能不能處理這個 bug黍少?” 而 A 表示:“我并不負責這一部分的內(nèi)容,你可以去問一下 B” 处面,接著測試人員就去問了B 厂置,B 也表示他不負責這塊,并讓測試去找C魂角,此時 C 表示他可以處理昵济,則責任到此結(jié)束。

iOS中責任鏈的使用

iOS中有許多情況使用到了責任鏈模式野揪,最典型的就是響應鏈访忿,一層層的往下傳遞事件,尋找可以處理事件的對象斯稳。

DEMO

公司處理人員申請經(jīng)費流程海铆,包括員工、經(jīng)理挣惰、CEO卧斟。首先定義經(jīng)費類和鏈式傳遞協(xié)議

// 鏈式協(xié)議
protocol Chain {
    var nextManagementLevel: Chain { get set } // 傳遞責任
    func shouldApproveExpenditure(expenditure: Expenditure)
}

// 經(jīng)費
class Expenditure {
    private var _amount: Int = 0
    var amount: Int { // 金額
        get {
            return _amount
        }
        set {
            _amount = newValue
        }
    }

    init(amount: Int) {
        self.amout = amout
    }
}

具體處理申請流程者

// 員工
class Employee: Chain {
    private var _nextManagementLevel: Chain?
    var nextManagementLevel: Chain {
        set {
            _nextManagementLevel = newValue
        }
        get {
            return _nextManagementLevel!
        }
    }

    func shouldApproveExpenditure(expenditure: Expenditure) {
        if expenditure.amount > 0 && expenditure.amount < 100 {
            print("An exployee can approve this expenditure")
        } else {
            if _nextManagementLevel != nil {
                nextManagementLevel.shouldApproveExpenditure(expenditure: expenditure)
            }
        }
    }
}

// 經(jīng)理
class Manager: Chain {
    private var _nextManagementLevel: Chain?
    var nextManagementLevel: Chain {
        set {
            _nextManagementLevel = newValue
        }
        get {
            return _nextManagementLevel!
        }
    }

    func shouldApproveExpenditure(expenditure: Expenditure) {
        if expenditure.amount > 101 && expenditure.amount < 1000 {
            print("Your manager can approve this expenditure")
        } else {
            if _nextManagementLevel != nil {
                nextManagementLevel.shouldApproveExpenditure(expenditure: expenditure)
            }
        }
    }
}

// CEO
class CEO: Chain {
    private var _nextManagementLevel: Chain?
    var nextManagementLevel: Chain {
        set {
            _nextManagementLevel = newValue
        }
        get {
            return _nextManagementLevel!
        }
    }

    func shouldApproveExpenditure(expenditure: Expenditure) {
        if expenditure.amount > 1001 && expenditure.amount < 10000 {
            print("The CEO can approve this expenditure")
        } else {
            print("This expenditure is too large and won't be approved")
        }
    }
}

使用

override func viewDidLoad() {
        super.viewDidLoad()

        // 創(chuàng)建執(zhí)行者
        let employee = Employee()
        let manager = Manager()
        let ceo = CEO()

        // 下一個處理者
        employee.nextManagementLevel = manager
        manager.nextManagementLevel = ceo

        // 申請經(jīng)費
        let expenditure = Expenditure(amount: 5)
        employee.shouldApproveExpenditure(expenditure: expenditure)

        // 修改申請金額
        expenditure.amount = 500
        employee.shouldApproveExpenditure(expenditure: expenditure)

        expenditure.amount = 5000
        employee.shouldApproveExpenditure(expenditure: expenditure)

        expenditure.amount = 50000
        employee.shouldApproveExpenditure(expenditure: expenditure)
    }

運行輸出

An exployee can approve this expenditure
Your boss can approve this expenditure
The CEO can approve this expenditure
This expenditure is too large and won't be approved

使用場景

  • 有多個對象可以處理同一個請求,動態(tài)確定誰可以處理該請求
  • 不明確指定接受者的情況下憎茂,給多個對象中的其中一個對象發(fā)送請求
  • 動態(tài)指定一組對象實現(xiàn)請求

雖然說責任鏈一般是指當一個對象可以處理時候停止傳遞珍语,但是在有需求的時候,就算不停止傳遞也沒有關系竖幔。

優(yōu)缺點

優(yōu)點

責任鏈模式非常顯著的優(yōu)點是將請求和處理分開板乙。請求者可以不用知道是誰處理的,處理者可以不用知道請求的全貌拳氢,兩者解耦募逞,提高系統(tǒng)的靈活性。

缺點

責任鏈模式有兩個非常顯著的缺點:一是性能問題饿幅,每個請求都是從鏈頭遍歷到鏈尾凡辱,特別是在鏈比較長的時候,性能是一共非常大的問題栗恩。二是調(diào)試不方便,特別是鏈條比較長洪燥,環(huán)節(jié)比較多的時候磕秤,由于采用了類似遞歸的方式,調(diào)試的時候邏輯可能比較復雜

參考

Design Patterns in Swift: Chain of Responsibility Pattern

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末捧韵,一起剝皮案震驚了整個濱河市市咆,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌再来,老刑警劉巖蒙兰,帶你破解...
    沈念sama閱讀 210,978評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件磷瘤,死亡現(xiàn)場離奇詭異,居然都是意外死亡搜变,警方通過查閱死者的電腦和手機采缚,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,954評論 2 384
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來挠他,“玉大人扳抽,你說我怎么就攤上這事≈城郑” “怎么了贸呢?”我有些...
    開封第一講書人閱讀 156,623評論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長拢军。 經(jīng)常有香客問我楞陷,道長,這世上最難降的妖魔是什么茉唉? 我笑而不...
    開封第一講書人閱讀 56,324評論 1 282
  • 正文 為了忘掉前任猜谚,我火速辦了婚禮,結(jié)果婚禮上赌渣,老公的妹妹穿的比我還像新娘魏铅。我一直安慰自己,他們只是感情好坚芜,可當我...
    茶點故事閱讀 65,390評論 5 384
  • 文/花漫 我一把揭開白布览芳。 她就那樣靜靜地躺著,像睡著了一般鸿竖。 火紅的嫁衣襯著肌膚如雪沧竟。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,741評論 1 289
  • 那天缚忧,我揣著相機與錄音悟泵,去河邊找鬼。 笑死闪水,一個胖子當著我的面吹牛糕非,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播球榆,決...
    沈念sama閱讀 38,892評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼朽肥,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了持钉?” 一聲冷哼從身側(cè)響起衡招,我...
    開封第一講書人閱讀 37,655評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎每强,沒想到半個月后始腾,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體州刽,經(jīng)...
    沈念sama閱讀 44,104評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,451評論 2 325
  • 正文 我和宋清朗相戀三年浪箭,在試婚紗的時候發(fā)現(xiàn)自己被綠了穗椅。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,569評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡山林,死狀恐怖房待,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情驼抹,我是刑警寧澤桑孩,帶...
    沈念sama閱讀 34,254評論 4 328
  • 正文 年R本政府宣布,位于F島的核電站框冀,受9級特大地震影響流椒,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜明也,卻給世界環(huán)境...
    茶點故事閱讀 39,834評論 3 312
  • 文/蒙蒙 一宣虾、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧温数,春花似錦绣硝、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,725評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至够傍,卻和暖如春甫菠,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背冕屯。 一陣腳步聲響...
    開封第一講書人閱讀 31,950評論 1 264
  • 我被黑心中介騙來泰國打工寂诱, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人安聘。 一個月前我還...
    沈念sama閱讀 46,260評論 2 360
  • 正文 我出身青樓痰洒,卻偏偏與公主長得像,于是被迫代替她去往敵國和親搞挣。 傳聞我的和親對象是個殘疾皇子带迟,可洞房花燭夜當晚...
    茶點故事閱讀 43,446評論 2 348

推薦閱讀更多精彩內(nèi)容

  • 1.責任鏈模式:避免將一個請求的發(fā)送者與接收者耦合在一起,讓多個對象都有機會處理請求囱桨,將接收請求的對象連接成一條鏈...
    CoderLWG閱讀 312評論 0 0
  • 一.什么是責任鏈模式? 責任鏈模式就是為一個請求或者一個動作創(chuàng)建一個接收者對象的鏈,這條鏈上的每一個對象都可以去響...
    LYSNote閱讀 3,656評論 0 7
  • 定義 為了避免請求發(fā)送者與接收者耦合在一起,讓多個對象都有可能接收請求嗅绰,將這些對象連接成一條鏈舍肠,并且沿著這條鏈傳遞...
    泥孩兒0107閱讀 300評論 0 0
  • 工廠模式類似于現(xiàn)實生活中的工廠可以產(chǎn)生大量相似的商品搀继,去做同樣的事情,實現(xiàn)同樣的效果;這時候需要使用工廠模式翠语。簡單...
    舟漁行舟閱讀 7,723評論 2 17
  • 設計模式概述 在學習面向?qū)ο笃叽笤O計原則時需要注意以下幾點:a) 高內(nèi)聚叽躯、低耦合和單一職能的“沖突”實際上,這兩者...
    彥幀閱讀 3,735評論 0 14