修仙的道路不需要敵人
Runtime漸漸清醒過來,畢竟兩瓶啤酒,沒有多大勁,它懷疑的眼光看的我很不自在。我說:“大兄弟,你喝醉的時候,都是我一直陪著你。”它問我:“趁我喝醉,你沒套我什么話吧?”以我的脾氣,況且現在我還知道了它的弱點,我當然不會慫:“講道理,套你話是必須的了。”我們對視了不到一秒,它笑著對我說:“哪有什么秘密,想知道什么,我都告訴你?!笨磥硭潜晃业奶孤收鞣?。
我買了一包辣條,沏了一壺茶,聽Runtime講述它的故事。
動態方法解析:
聽Runtime講,有時候會遇到@dynamic propertyName;
這種招式,破解方法就是通過resolveInstanceMethod
動態的提供屬性的相關方法。一個OC方法調用,至少有self和_cmd這兩個參數,模版大概如下:
void dynamicMethodIMP(id self, SEL _cmd) {
// implementation....
}
然后這樣使用resolveInstanceMethod
:
+ (BOOL)resolveInstanceMethod:(SEL)aSEL
{
if (aSEL == @selector(needDynamicCallMethod)) {
class_addMethod([self class], aSEL, (IMP) dynamicMethodIMP, "v@:");
return YES;
}
return [super resolveInstanceMethod:aSEL];
}
一個類在消息轉發機制開始之前是有機會去做動態方法解析的。也就是說在調用了一個未知的方法后,我們可以像上面的代碼那樣,在resolveInstanceMethod:
中給類添加這個未知方法,使方法的調用順利完成。如果在resolveInstanceMethod:
中返回了NO,那么將啟動消息轉發機制。
注: 這種方案的前提是dynamicMethodIMP
函數已經被提前定義。
消息轉發:
“曾今有一份真摯的消息擺在我面前,我沒有珍惜,等程序崩潰的時候,我才后悔莫及。人世間最痛苦的事莫過于此。如果上天能夠給我再來一次的機會,我一定對它說‘臭傻嗨’。如果一定要給這三個字加上期限的話,我希望是app被拒之后?!盧untime感慨萬千的說了這些騷話,我瞬間精神抖擻,聽它講訴它與消息之間的八卦。
向一個對象發送一個它不能處理的消息是錯誤的,但是,在錯誤發生之前,Runtime系統會給此對象第二次處理這個消息的機會。
- 重寫
forwardingTargetForSelector:
方法。返回值可以是一個能夠處理此消息的對象,或者是nil。 - 重寫
forwardInvocation:
方法。參數NSInvocation對象包含了消息的接收者、選擇器、參數和返回值。
注: 在內存管理方面,這個類不保留默認包含的調用的參數,我們可以使用retainArguments
方法破招。
重寫forwardingTargetForSelector:
方法并返回一個對象,這樣做可以模擬出多重繼承的效果。這里著重說一下轉發機制與多重繼承的重要區別:多重繼承在一個對象中組合了不同的能力,它傾向于大的,多方面的對象;轉發機制是給不同的對象分配不同的職責,它將問題分解成更小的對象,并把這些對象以一種透明的方式與消息發送者關聯。
- (id)forwardingTargetForSelector:(SEL)aSelector {
if ([otherObject respondsToSelector:aSelector]) {
return otherObject;
}
return nil; //return [super forwardingTargetForSelector:aSelector];
}
如果希望對象能夠像正常繼承那樣正確返回,你就需要重新實現 respondsToSelector:
方法,例如:
- (BOOL)respondsToSelector:(SEL)aSelector
{
if ( [super respondsToSelector:aSelector] )
return YES;
else {
/* 如果消息能夠通過轉發機制完成, 就返回YES */
}
return NO;
}
重寫forwardInvocation:
方法的同時,也必須重寫methodSignatureForSelector:
方法,因為需要將方法的描述(簽名)告知響應消息的代理對象。
- (NSMethodSignature *)methodSignatureForSelector:(SEL)selector
{
NSMethodSignature *signature = [super methodSignatureForSelector:selector];
if (!signature) {
signature = [surrogate methodSignatureForSelector:selector];
}
return signature;
}
// 這里的invocation就是通過返回的signature初始化來的
- (void)forwardInvocation:(NSInvocation *)invocation
{
SEL aSelector = [invocation selector];
if ([friend respondsToSelector:aSelector])
[invocation invokeWithTarget:friend];
else
[super forwardInvocation:invocation];
}
一包辣條吃完,Runtime就給我透漏了這么多招式,我頓時感慨萬千:這要是靠套話的方式去了解它們,我得花多少錢買酒啊。
關注微信公眾號CodingArtist,可以第一時間得到文章更新通知! _