iOS 使用 NSProxy 實現(xiàn)消息分發(fā)器

??最近準備進一步重構(gòu)某幾個頁面怕品,從結(jié)構(gòu)上講用的是 MVVM,較為清晰明了句各,同時也不至于所有代碼都集中在 UIViewController 里導(dǎo)致一團麻昆码,但是隨著一個頁面功能的更改替換,還有業(yè)務(wù)統(tǒng)計代碼等等的加入锰蓬,終究還是忍不住想要稍微整理整理。

??由于頁面多為 UITableView 且頁面多復(fù)用眯漩,各種 統(tǒng)計 或者 頁面差異展示芹扭,所以想如果可以多個代理均可按先后順序執(zhí)行,那就可以將一些 統(tǒng)計 或者 頁面差異展示 跟其它正常業(yè)務(wù)進行分離赦抖,提升項目代碼的可維護性舱卡。

??其實上面說的就是 透明的分布式消息傳遞 也可稱為 消息分發(fā)器 ,所以在這里要介紹一下非 NSObject 的子類队萤,而是另一個基類 NSProxy轮锥,從命名上看,就是代理要尔,即設(shè)計模式中的 代理模式舍杜,NSProxy 是一個抽象超類,實現(xiàn)根類要求的基礎(chǔ)方法赵辕,包括<NSObject>協(xié)議中定義的方法既绩,可作為其他對象替身(甚至不存在的對象),NSProxy 負責(zé)把消息轉(zhuǎn)發(fā)給真正的 target 的代理類还惠,利用這個特性饲握,NSProxy 就能 透明的分布式消息傳遞

  • NSProxy

??從蘋果官方文檔的介紹可以看到,他就是一個替代蚕键,幫忙轉(zhuǎn)發(fā)消息給具體實現(xiàn)的類救欧,或者懶加載一個較大的實例

Typically, a message to a proxy is forwarded to the real object or causes the proxy to load (or transform itself into) the real object

??作為抽象類不能直接使用 NSProxy ,它不提供初始化的方法锣光,并且在接收到它沒有響應(yīng)的任何消息時引發(fā)異常笆怠,因此只能繼承并在具體的子類提供初始化或創(chuàng)建方法,并實現(xiàn) forwardInvocation:methodSignatureForSelector:

??所以其實只要實現(xiàn)了這兩個方法即可誊爹,至于其它的一些方法骑疆,未實現(xiàn)均會走到消息轉(zhuǎn)發(fā)這一步田篇,若是實現(xiàn)其它譬如respondsToSelector:也無妨

  • NSProxyNSObject 的消息傳遞的不同
    ??NSObject 收到消息會先去緩存列表查找SEL,若是找不到箍铭,就到自身方法列表中查找泊柬,依然找不到就順著繼承鏈進行查找,依然找不到的話诈火,那就是 unknown selector兽赁,進入消息轉(zhuǎn)發(fā)程序
    ??1. +(BOOL)resolveInstanceMethod: 其返回值為Boolean類型,表示這個類是否通過class_addMethod新增一個實例方法用以處理該 unknown selector冷守,也就是說在這里可以新增一個處理 unknown selector的方法刀崖,若不能,則繼續(xù)往下傳遞(若 unknown selector是類方法拍摇,那么調(diào)用 +(BOOL)resolveClassMethod:
    ??2. - (id)forwardingTargetForSelector: 這是第二次機會進行處理 unknown selector亮钦,即轉(zhuǎn)移給一個能處理 unknown selector的其它對象,若返回一個其它的執(zhí)行對象充活,那消息從 id objc_msgSend ( id self, SEL op, ...) 重新開始蜂莉,若不能,則返回 nil混卵,并繼續(xù)向下傳遞映穗,最后的一次消息處理機會(3 與 4 配套)
    ??3. - (NSMethodSignature *)methodSignatureForSelector: 返回攜帶參數(shù)類型、返回值類型和長度等的 selector 簽名信息 NSMethodSignature對象幕随,Runtime 內(nèi)部會基于 NSMethodSignature 實例構(gòu)建一個NSInvocation 對象蚁滋,作為回調(diào)- (void)forwardInvocation:的入?yún)?br> ??4. - (void)forwardInvocation:這一步可以對傳進來的 NSInvocation 進行一些操作,它主要在對象之間或者應(yīng)用程序之間存儲和轉(zhuǎn)發(fā)消息(命令模式的實現(xiàn))赘淮,靈活性很高辕录,譬如修改 target 、參數(shù)梢卸、甚至返回值踏拜,有興趣可以去了解下NSInvocation
    ??對于 NSProxy 就沒有這么復(fù)雜了,接收到 unknown selector 后低剔,直接回調(diào)- (NSMethodSignature *)methodSignatureForSelector:- (void)forwardInvocation:, 消息轉(zhuǎn)發(fā)過程簡單的很多

  • 消息分發(fā)器
    • 基于以上速梗,實現(xiàn)上也很簡單,就是將具體接收消息的實際對象保存在容器中襟齿,并按順序讓它們接收消息即可
/**
 Normal forwarding 的第一步姻锁,也是消息轉(zhuǎn)發(fā)的最后一次機會--這個針對NSObject, 對于 NSProxy 未實現(xiàn)立馬走這里
 消息獲得函數(shù)的參數(shù)和返回值類型,即返回一個函數(shù)簽名

 @param sel selector 方法選擇子
 @return NSMethodSignature 函數(shù)簽名
         返回nil猜欺,Runtime 則會發(fā)出 -doesNotRecognizeSelector: 消息位隶,程序 crash
         返回了NSMethodSignature,Runtime 就會創(chuàng)建一個 NSInvocation 對象并發(fā)送 -forwardInvocation: 消息給目標對象
 */
- (nullable NSMethodSignature *)methodSignatureForSelector:(SEL)sel {
    for (id obj in self.targets) {
        if ([obj respondsToSelector:sel]) {
            return [obj methodSignatureForSelector:sel];
        }
    }
    return [super methodSignatureForSelector:sel];
}
/**
可以在 -forwardInvocation: 里修改傳進來的 NSInvocation 對象开皿,然后發(fā)送 -invokeWithTarget: 消息給它涧黄,傳進去一新的目標執(zhí)行

 @param invocation 對一個消息的描述篮昧,包括 selector 以及參數(shù)等信息
 */
- (void)forwardInvocation:(NSInvocation *)invocation {
    BOOL invoked = NO;
    for (id obj in self.targets) {
        if ([obj respondsToSelector:invocation.selector]) {
            [invocation invokeWithTarget:obj];
            invoked = YES;
        }
    }
    if (!invoked) {
        [super forwardInvocation:invocation];
    }
}
  • 使用

?a. 偽多繼承

// dispathProx 同時可以執(zhí)行 LYAnimal 和 LYPerson 的方法,看起來像是多繼承笋妥。懊昨。。其實假的啦
  id dispathProx = ly_dispatchProxy([LYAnimal new], [LYPerson new], nil);
  [dispathProx talk];
  [dispathProx walk];
  [dispathProx pushers];
  [dispathProx shovel];

?b.協(xié)議多分發(fā) 即 消息分發(fā)用的最多的一種

  // ?? self 持有 delegateProxy春宣,delegateProxy 內(nèi)部持有傳入的消息接收對象酵颁,里面也有 self, 所以使用 ly_weakObject(self),改為 weak 引用
  // ?? 當有返回值時月帝,后面的返回值會覆蓋前面的
  self.delegateProxy = (LYDispatchProxy <UITableViewDataSource, UITableViewDelegate>*)ly_dispatchProxy([LYTableDelegate new], ly_weakObject(self), nil);
  // 可在LYTableDelegate實現(xiàn)一些統(tǒng)計業(yè)務(wù)或者差異展示的設(shè)置處理
   ....
   
  talbe.delegate = self.delegateProxy;
  talbe.dataSource = self.delegateProxy;
  // 或者也可以如下用法躏惋,憑想象吧。嚷辅。簿姨。其實肚子餓了,所以咕咕叫簸搞,不想想了
   LYDispatchProxy *btnTargetProxy = ly_dispatchProxy(XXX0, XXX1, nil);
   [btn addTarget:btnTargetProxy action:NSSelectorFromString(@"xxxx") forControlEvents:UIControlEventTouchUpInside];

??業(yè)務(wù)做的多了扁位,慢慢也有了更多的思考,但是無論是想什么攘乒,偶現(xiàn)在就是要吃燒烤,別的都不管 哼 哼 哼

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末惋鹅,一起剝皮案震驚了整個濱河市则酝,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌闰集,老刑警劉巖沽讹,帶你破解...
    沈念sama閱讀 216,470評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異武鲁,居然都是意外死亡爽雄,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,393評論 3 392
  • 文/潘曉璐 我一進店門沐鼠,熙熙樓的掌柜王于貴愁眉苦臉地迎上來挚瘟,“玉大人,你說我怎么就攤上這事饲梭〕烁牵” “怎么了?”我有些...
    開封第一講書人閱讀 162,577評論 0 353
  • 文/不壞的土叔 我叫張陵憔涉,是天一觀的道長订框。 經(jīng)常有香客問我,道長兜叨,這世上最難降的妖魔是什么穿扳? 我笑而不...
    開封第一講書人閱讀 58,176評論 1 292
  • 正文 為了忘掉前任衩侥,我火速辦了婚禮殴蓬,結(jié)果婚禮上仅讽,老公的妹妹穿的比我還像新娘。我一直安慰自己割捅,他們只是感情好泽谨,可當我...
    茶點故事閱讀 67,189評論 6 388
  • 文/花漫 我一把揭開白布璧榄。 她就那樣靜靜地躺著,像睡著了一般吧雹。 火紅的嫁衣襯著肌膚如雪骨杂。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,155評論 1 299
  • 那天雄卷,我揣著相機與錄音搓蚪,去河邊找鬼。 笑死丁鹉,一個胖子當著我的面吹牛妒潭,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播揣钦,決...
    沈念sama閱讀 40,041評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼雳灾,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了冯凹?” 一聲冷哼從身側(cè)響起谎亩,我...
    開封第一講書人閱讀 38,903評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎宇姚,沒想到半個月后匈庭,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,319評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡浑劳,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,539評論 2 332
  • 正文 我和宋清朗相戀三年阱持,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片魔熏。...
    茶點故事閱讀 39,703評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡衷咽,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出蒜绽,到底是詐尸還是另有隱情兵罢,我是刑警寧澤,帶...
    沈念sama閱讀 35,417評論 5 343
  • 正文 年R本政府宣布滓窍,位于F島的核電站卖词,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜此蜈,卻給世界環(huán)境...
    茶點故事閱讀 41,013評論 3 325
  • 文/蒙蒙 一即横、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧裆赵,春花似錦东囚、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,664評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至植兰,卻和暖如春份帐,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背楣导。 一陣腳步聲響...
    開封第一講書人閱讀 32,818評論 1 269
  • 我被黑心中介騙來泰國打工废境, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人筒繁。 一個月前我還...
    沈念sama閱讀 47,711評論 2 368
  • 正文 我出身青樓噩凹,卻偏偏與公主長得像,于是被迫代替她去往敵國和親毡咏。 傳聞我的和親對象是個殘疾皇子驮宴,可洞房花燭夜當晚...
    茶點故事閱讀 44,601評論 2 353

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