Objective-C Method Swizzling增強注釋版。有些代碼可能拿過來能直接用,但是我們并不知道其原理或者理解想法庆寺,所以總是一知半解走净。昨天看了下Objective-C Method Swizzling,這是很典型的案例邮旷,因為很繞黄选!(只有我覺得繞嗎?看著看著自己就懵了[手動一臉懵逼]),所以自己加了點注釋办陷,幫助下理解(可能理解的不對貌夕,大家不吝賜教吧!)
我理解的哈~:重點就是方法名和方法的實現(xiàn) 我們要區(qū)別開 - (void)...; 是方法的聲明 - (void)...{}這個是方法的實現(xiàn)民镜,交換方法的原理其實就是維持方法申明不變啡专,然后交換兩個方法的實現(xiàn),引起的結果就是 調用a方法:[... a]制圈,而實際上調用的是b方法的實現(xiàn):-b{}们童,反之亦然。
+ (void)load {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
Class class = [self class]; //替換的是對象(實例)方法
//!重點強調下 如果替換的是類方法 格式應該是
Class class = object_getClass[self class];
//? ? ? ? SEL originalSelector = @selector(viewWillAppear:);
//? ? ? ? SEL swizzledSelector = @selector(mrc_viewWillAppear:);
????? SEL originalSelector = @selector(log);
????? SEL swizzledSelector = @selector(customLog);
//對象方法替換
????? Method originalMethod = class_getInstanceMethod(class, originalSelector);
????? Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
//類方法替換
????? Method originalClassMethod = class_getClassMethod(class, originalSelector);
????? Method swizzledClassMethod = class_getClassMethod(class, swizzledSelector);
///----------------------------------------
/// @name 最核心需要理解的代碼
///----------------------------------------
BOOL success =
class_addMethod(class, originalSelector, method_getImplementation(swizzledMethod), method_getTypeEncoding(swizzledMethod));
//? ? ? ? class_addMethod 為originalSelector方法添加實現(xiàn) 成功:說明該方法沒有在.m中實現(xiàn) 于是就把swizzledMethod的實現(xiàn)添加給了originalSelector? 失斁小:說明該方法有在.m中實現(xiàn)了 那么就將兩個方法的實現(xiàn)進行替換 注意 是替換實現(xiàn) 而不是方法名
//Method 組成是個結構體 成分:SELname:方法名; constchar*types:參數(shù); IMPimp:實現(xiàn);
//class_addMethod照我的理解 就是添加一個結構體慧库,然后給SEL和IMPimp賦值,所以這里其實也是給originalSelector替換好了實現(xiàn)(成功success=yes)
//success 作用:
//如果返回YES 說明這個類.m沒有實現(xiàn)了這個方法(那么肯定是父類實現(xiàn)了)這里我之前有個疑問亥鬓,如果originalMethod
一直沒有實現(xiàn)完沪,或者都不存在呢?那下面的class_replaceMethod導致的后果難道不是奔潰嗎嵌戈?因為你用了一個不存在的實現(xiàn)去替換另一個方法的聲明覆积,一調用不是炸了?后來我想了想熟呛,originalMethod不存在我們也沒法調用對不對宽档,替換了有怎么樣?我們沒法去調用呀庵朝!originalMethod只聲明不實現(xiàn)呢吗冤?那肯定奔潰呀!這不很正常么~"茅廁頓開"
if (success) {
????? class_replaceMethod(class, swizzledSelector, method_getImplementation(originalMethod), method_getTypeEncoding(originalMethod));
}
//如果返回NO 說明這個類.m實現(xiàn)了這個方法
else {
????? method_exchangeImplementations(originalMethod, swizzledMethod);//交換兩個方法的實現(xiàn)
//e.g 調mrc_viewWillAppear方法九府,聯(lián)動的是viewWillAppear的實現(xiàn)
//調viewWillAppear方法椎瘟,聯(lián)動的是的實現(xiàn)mrc_viewWillAppear
}
????? NSLog(@"loadReady");
});
}
#pragma mark - Method Swizzling
//重點在這,乍一看不是循環(huán)調用了嗎侄旬?要記住一點 方法的聲明和方法的調用不是一個概念
- (void)customLog{//這是方法的實現(xiàn)
? ? ? [self customLog];//這是通過方法的聲明調用對應方法的實現(xiàn) 而在這里customLog方法的實現(xiàn)已經(jīng)被我們替換掉了
?????? NSLog(@"====");
}