iOS黑魔法之method_exchangeImplementations

Objective-C黑魔法使用适当能给编码带来很大的便利,Swizzling就是其中之一。比如集成友盟统计时,如果按照常规方法来做的话,需要在每个页面打点,页面多多话,这不搞死人吗?有没有一个简便的方法能够一劳永逸尼,答案就是Swizzling。利用Objective-C的动态特性,在运行时把原本selector对应的实现绑定到我们指定的实现来。

一.应用场景:

1.数组越界判断
 2.可变字典插入空元素
 3.集成友盟统计时,不必在每个控制中添加代码

二.到底怎么用:

每一个类都有对应的类方法列表,以及实例方法列表,selector的名字和方法实现是一一对应的关系,IMP类似函数指针,每个selector都对应一个IMP。如下图:

0ECADE10-EDF2-4381-96E0-A87E1752815C.png

 在 <objc/runtime.h> 中有一个method_exchangeImplementations方法,可以改变selector指向的IMP',,说白了,我们就是要改变selector的实现。比如在友盟统计中,我们需要在 - (void)viewWillAppear:(BOOL)animated 中打点。其实我们可以把打点的代码写在父类中,然后让需要打点的页面都继承这个父类,但是工作量就比较大,而且代码恶心。
 最优解就是我们定义一个Category,在这个Category中,偷偷- (void)viewWillAppear:(BOOL)animated中的实现指向另一个我们预想的IMP。

3BB852A3-6BB6-4645-B954-7715E42AA9A7.png

我们必须在方法没有执行之前把它的实现替换掉,否则就没有意义了,那么在那个时机替换实现尼?'+(void)load'方法APP启动前就被调用了,并且它在整个程序生命周期里只执行一次,所以我们可以在这里搞点小动作。我们可以打断点验证一下的,新建一个iOS工程,分别在

+(void)load,
int main(int argc, char * argv[]) {},
- (BOOL)application:(UIApplication *)application   didFinishLaunchingWithOptions:(NSDictionary *)launchOptions

这三个方法里打断点,发现程序是按照-->load-->main-->didFinishLaunchingWithOptions:的顺序来执行的,也是说我们的把替换代码写在+(void)load中是正确的。

废话少说,直接上代码
#import "UIViewController+Swizzling.h"
#import <objc/runtime.h>
@implementation UIViewController (Swizzling)

+(void)load{
    //虽然load只执行一次,但是为了保险起见,我们还是给加个dispatch_once吧,良好的编程习惯,从这里开始
    static dispatch_once_t token;
    dispatch_once(&token, ^{
        
        SEL orginSel = @selector(viewWillAppear:);
        SEL overrideSel = @selector(overrideViewWillAppear:);
        
        Method originMethod = class_getInstanceMethod([self class], orginSel);
        Method overrideMethod = class_getInstanceMethod([self class], overrideSel);
        
        //原来的类没有实现指定的方法,那么我们就得先做判断,把方法添加进去,然后进行替换
        if (class_addMethod([self class], orginSel, method_getImplementation(overrideMethod) , method_getTypeEncoding(originMethod))) {
            class_replaceMethod([self class],
                                overrideSel,
                                method_getImplementation(originMethod),
                                method_getTypeEncoding(originMethod));
            
        }else{
            //交换实现
            method_exchangeImplementations(originMethod, overrideMethod);
        }
    });
}

 - (void)overrideViewWillAppear:(BOOL)animation{
    NSLog(@"%@-----overrideViewWillAppear",self);
    //这里并不会造成死循环,因为这个时候是去调用原来的ViewWillAppear:(BOOL)animation方法了。
    [self overrideViewWillAppear:animation];
}

借用这个图来说明一下为什么-(void)overrideViewWillAppear:(BOOL)animation里调用自身不会死循环。


1122433-708ca3964274b58a.jpg

-viewWillAppear (SEL) -> -overrideViewWillAppear (IMP)-> [self overrideViewWillAppear:animation] (SEL) -> -viewWillAppear(IMP)
 这个流程就一目了然了,当页面将要出现时调用-viewWillAppear,但是这个方法的实现已经被我们换了,最终会掉到我们调前写好的方法-overrideViewWillAppear,但是我们通过self来调用-overrideViewWillAppear时,却又走的是-viewWillAppear,所以不会出现死循环。
 所以最终我们在overrideViewWillAppear里面添加了打点得代码,并且还能不影响已经在-viewWillAppear添加的代码。

另外一个iOSer的痛点就是,我们在给可变字典添加元素时,一不小心就奔溃了,额,也是挺奔溃的啊~。其中的一个原因是添加到字典的value为nil了。刚好我们可以用刚刚学的的钩子,解决这个问题。思路就是每次调用 setObject:forKey:的时候,我们偷偷的得对Object做一个非空判断,如果为空就不给添加到字典里面来。
思路有了,那开始码代码了。

#import "NSMutableDictionary+Swizzling.h"
#import <objc/runtime.h>

@implementation NSMutableDictionary (Swizzling)

+ (void)load
{
    static dispatch_once_t token;
    dispatch_once(&token, ^{
        
        SEL orginSel = @selector(setObject:forKey:);
        SEL overrideSel = @selector(overrideSetObject:forKey:);
        
        Method originMethod = class_getInstanceMethod([self class], orginSel);
        Method overrideMethod = class_getInstanceMethod([self class], overrideSel);
        
        //原来的类没有实现指定的方法,那么我们就得先做判断,把方法添加进去,然后进行替换
        if (class_addMethod([self class], orginSel, method_getImplementation(overrideMethod) , method_getTypeEncoding(originMethod))) {
            class_replaceMethod([self class],
                                overrideSel,
                                method_getImplementation(originMethod),
                                method_getTypeEncoding(originMethod));
            
        }else{
            //交换实现
            method_exchangeImplementations(originMethod, overrideMethod);
        }
    });
}

- (void)overrideSetObject:(id)anObject forKey:(id <NSCopying>)aKey;
{
    if (anObject) {
        NSLog(@"%@--overrideSetObject",self);
        /** 注意:必须调用自己的方法名 */
        [self overrideSetObject:anObject forKey:aKey];
    }
}
@end

我们使用一下这个分类试试

 NSMutableDictionary *dic = [[NSMutableDictionaryalloc]init];
[dic setObject:@"testObject"forKey:@"myKey"];

结果发现- (void)overrideSetObject:(id)anObject forKey:(id <NSCopying>)aKey这个方法根本就没有被调用。原因何在尼?原来NSMutableDictionary是一个族类(关于族类建议参考《Effective Objective-C 2.0 编写高质量iOS与OS X代码的52个有效方法》第九条),我们代码里通过[selfclass]返回的是当前类的类名,也就是NSMutableDictionary,而实际上应该是__NSDictionaryM (dic __NSDictionaryM * 1 key/value pair 0x00007f84a4f0daf0)(在控制台中po一下这个dic就能看出来了,这个dic的类型是__NSDictionaryM)。所以我把以上代码改一下:
+ (void)load
{
static dispatch_once_t token;
dispatch_once(&token, ^{

        SEL orginSel = @selector(setObject:forKey:);
        SEL overrideSel = @selector(overrideSetObject:forKey:);
        
        Class o_class = objc_getClass("__NSDictionaryM");
        Method originMethod = class_getInstanceMethod(o_class, orginSel);
        Method overrideMethod = class_getInstanceMethod(o_class, overrideSel);
        
        //原来的类没有实现指定的方法,那么我们就得先做判断,把方法添加进去,然后进行替换
        if (class_addMethod(o_class, orginSel, method_getImplementation(overrideMethod) , method_getTypeEncoding(originMethod))) {
            class_replaceMethod(o_class,
                                overrideSel,
                                method_getImplementation(originMethod),
                                method_getTypeEncoding(originMethod));
            
        }else{
            //交换实现
            method_exchangeImplementations(originMethod, overrideMethod);
        }
    });
}

run一下发行可以了。
 在cocoa框架里面数组和字典这样的族类,最终初始化出来的实例的类型,并不是预想的,下图才是它们的’真身’

类                               “真身”

NSArray                       __NSArrayI

NSMutableArray                __NSArrayM

NSDictionary                  __NSDictionaryI

NSMutableDictionary           __NSDictionaryM
三.不足

在程序启动的时候,系统也会多次调用setObject:forKey:方法,从而也会调用我们写的那个钩子,所以感觉还不算完善,毕竟系统调用的,我们管不着了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,709评论 0 9
  • 继上Runtime梳理(四) 通过前面的学习,我们了解到Objective-C的动态特性:Objective-C不...
    小名一峰阅读 752评论 0 3
  • 目录 Objective-C Runtime到底是什么 Objective-C的元素认知 Runtime详解 应用...
    Ryan___阅读 1,938评论 1 3
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,191评论 0 7
  • 1 我最常翻看的相册中 总有一个光彩照人的笑影, 那一段时光,总有她的出现, 每天与她相见, 就像参加了一场场精彩...
    衣饰忆流年阅读 340评论 0 4