最近準(zhǔn)備進(jìn)一步重構(gòu)某幾個(gè)頁(yè)面哑梳,從結(jié)構(gòu)上講用的是MVVM劲阎,較為清晰明了,同時(shí)也不至于所有代碼都集中在UIViewController里導(dǎo)致一團(tuán)麻鸠真,但是隨著一個(gè)頁(yè)面功能的更改替換悯仙,還有業(yè)務(wù)統(tǒng)計(jì)代碼等等的加入,終究還是忍不住想要稍微整理整理吠卷。
由于頁(yè)面多為UITableView且頁(yè)面多復(fù)用锡垄,各種統(tǒng)計(jì)或者頁(yè)面差異展示,所以想如果可以多個(gè)代理均可按先后順序執(zhí)行祭隔,那就可以將一些統(tǒng)計(jì)或者頁(yè)面差異展示跟其它正常業(yè)務(wù)進(jìn)行分離货岭,提升項(xiàng)目代碼的可維護(hù)性。
其實(shí)上面說(shuō)的就是透明的分布式消息傳遞也可稱為消息分發(fā)器疾渴,所以在這里要介紹一下非NSObject的子類茴她,而是另一個(gè)基類NSProxy,從命名上看程奠,就是代理丈牢,即設(shè)計(jì)模式中的代理模式,NSProxy是一個(gè)抽象超類瞄沙,實(shí)現(xiàn)根類要求的基礎(chǔ)方法己沛,包括<NSObject>協(xié)議中定義的方法,可作為其他對(duì)象替身(甚至不存在的對(duì)象)距境,NSProxy負(fù)責(zé)把消息轉(zhuǎn)發(fā)給真正的target的代理類申尼,利用這個(gè)特性,NSProxy就能透明的分布式消息傳遞
NSProxy
從蘋果官方文檔的介紹可以看到垫桂,他就是一個(gè)替代师幕,幫忙轉(zhuǎn)發(fā)消息給具體實(shí)現(xiàn)的類,或者懶加載一個(gè)較大的實(shí)例
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)的任何消息時(shí)引發(fā)異常灭将,因此只能繼承并在具體的子類提供初始化或創(chuàng)建方法,并實(shí)現(xiàn)forwardInvocation:和methodSignatureForSelector:
所以其實(shí)只要實(shí)現(xiàn)了這兩個(gè)方法即可后控,至于其它的一些方法庙曙,未實(shí)現(xiàn)均會(huì)走到消息轉(zhuǎn)發(fā)這一步,若是實(shí)現(xiàn)其它譬如respondsToSelector:也無(wú)妨
NSProxy與NSObject的消息傳遞的不同
NSObject收到消息會(huì)先去緩存列表查找SEL浩淘,若是找不到捌朴,就到自身方法列表中查找,依然找不到就順著繼承鏈進(jìn)行查找张抄,依然找不到的話砂蔽,那就是unknown selector,進(jìn)入消息轉(zhuǎn)發(fā)程序
1.+(BOOL)resolveInstanceMethod:其返回值為Boolean類型署惯,表示這個(gè)類是否通過(guò)class_addMethod新增一個(gè)實(shí)例方法用以處理該unknown selector左驾,也就是說(shuō)在這里可以新增一個(gè)處理unknown selector的方法,若不能泽台,則繼續(xù)往下傳遞(若unknown selector是類方法,那么調(diào)用+(BOOL)resolveClassMethod:)
2.- (id)forwardingTargetForSelector:這是第二次機(jī)會(huì)進(jìn)行處理unknown selector矾缓,即轉(zhuǎn)移給一個(gè)能處理unknown selector的其它對(duì)象怀酷,若返回一個(gè)其它的執(zhí)行對(duì)象,那消息從id objc_msgSend ( id self, SEL op, ...)重新開始嗜闻,若不能蜕依,則返回nil,并繼續(xù)向下傳遞琉雳,最后的一次消息處理機(jī)會(huì)(3 與 4 配套)
3.- (NSMethodSignature *)methodSignatureForSelector:返回?cái)y帶參數(shù)類型样眠、返回值類型和長(zhǎng)度等的selector簽名信息NSMethodSignature對(duì)象,Runtime內(nèi)部會(huì)基于NSMethodSignature實(shí)例構(gòu)建一個(gè)NSInvocation對(duì)象翠肘,作為回調(diào)- (void)forwardInvocation:的入?yún)?/p>
4.- (void)forwardInvocation:這一步可以對(duì)傳進(jìn)來(lái)的NSInvocation進(jìn)行一些操作檐束,它主要在對(duì)象之間或者應(yīng)用程序之間存儲(chǔ)和轉(zhuǎn)發(fā)消息(命令模式的實(shí)現(xiàn)),靈活性很高束倍,譬如修改target被丧、參數(shù)、甚至返回值绪妹,有興趣可以去了解下NSInvocation
對(duì)于NSProxy就沒有這么復(fù)雜了甥桂,接收到unknown selector后,直接回調(diào)- (NSMethodSignature *)methodSignatureForSelector:和- (void)forwardInvocation:, 消息轉(zhuǎn)發(fā)過(guò)程簡(jiǎn)單的很多
基于以上邮旷,實(shí)現(xiàn)上也很簡(jiǎn)單黄选,就是將具體接收消息的實(shí)際對(duì)象保存在容器中,并按順序讓它們接收消息即可
/**
Normal forwarding 的第一步婶肩,也是消息轉(zhuǎn)發(fā)的最后一次機(jī)會(huì)--這個(gè)針對(duì)NSObject, 對(duì)于 NSProxy 未實(shí)現(xiàn)立馬走這里
消息獲得函數(shù)的參數(shù)和返回值類型办陷,即返回一個(gè)函數(shù)簽名
@param sel selector 方法選擇子
@return NSMethodSignature 函數(shù)簽名
? ? ? ? 返回nil貌夕,Runtime 則會(huì)發(fā)出 -doesNotRecognizeSelector: 消息,程序 crash
? ? ? ? 返回了NSMethodSignature懂诗,Runtime 就會(huì)創(chuàng)建一個(gè) NSInvocation 對(duì)象并發(fā)送 -forwardInvocation: 消息給目標(biāo)對(duì)象
*/-(nullable NSMethodSignature*)methodSignatureForSelector:(SEL)sel{for(id objinself.targets){if([obj respondsToSelector:sel]){return[obj methodSignatureForSelector:sel];}}return[supermethodSignatureForSelector:sel];}
/**
可以在 -forwardInvocation: 里修改傳進(jìn)來(lái)的 NSInvocation 對(duì)象蜂嗽,然后發(fā)送 -invokeWithTarget: 消息給它,傳進(jìn)去一新的目標(biāo)執(zhí)行
@param invocation 對(duì)一個(gè)消息的描述殃恒,包括 selector 以及參數(shù)等信息
*/-(void)forwardInvocation:(NSInvocation*)invocation{BOOL invoked=NO;for(id objinself.targets){if([obj respondsToSelector:invocation.selector]){[invocation invokeWithTarget:obj];invoked=YES;}}if(!invoked){[superforwardInvocation:invocation];}}
使用
?a. 偽多繼承
// dispathProx 同時(shí)可以執(zhí)行 LYAnimal 和 LYPerson 的方法植旧,看起來(lái)像是多繼承。离唐。病附。其實(shí)假的啦id dispathProx=ly_dispatchProxy([LYAnimalnew],[LYPersonnew],nil);[dispathProx talk];[dispathProx walk];[dispathProx pushers];[dispathProx shovel];
?b.協(xié)議多分發(fā) 即 消息分發(fā)用的最多的一種
// ?? self 持有 delegateProxy,delegateProxy 內(nèi)部持有傳入的消息接收對(duì)象亥鬓,里面也有 self, 所以使用 ly_weakObject(self)完沪,改為 weak 引用// ?? 當(dāng)有返回值時(shí),后面的返回值會(huì)覆蓋前面的self.delegateProxy=(LYDispatchProxy<UITableViewDataSource,UITableViewDelegate>*)ly_dispatchProxy([LYTableDelegate new],ly_weakObject(self),nil);// 可在LYTableDelegate實(shí)現(xiàn)一些統(tǒng)計(jì)業(yè)務(wù)或者差異展示的設(shè)置處理....talbe.delegate=self.delegateProxy;talbe.dataSource=self.delegateProxy;
// 或者也可以如下用法嵌戈,憑想象吧覆积。。熟呛。其實(shí)肚子餓了宽档,所以咕咕叫,不想想了LYDispatchProxy*btnTargetProxy=ly_dispatchProxy(XXX0,XXX1,nil);[btn addTarget:btnTargetProxy action:NSSelectorFromString(@"xxxx")forControlEvents:UIControlEventTouchUpInside]