OC中swizzling的“移魂大法”

前言

OC的runtime机制是一个很有意思的东西,今天就来讲讲OC方法实现的交换,当然称之为“移魂大法”确实有点夸张了,不过它真的是一个很神奇的东西。

交换的两种方式

  1. 在本类中进行方法交换

先贴上代码:

    voidMethodSwizzle(Class c,SEL origSEL,SEL overrideSEL)
    {
        Method origMethod = class_getInstanceMethod(c, origSEL);
        Method overrideMethod= class_getInstanceMethod(c, overrideSEL);
        if(class_addMethod(c, origSEL, method_getImplementation(overrideMethod),method_getTypeEncoding(overrideMethod)))
        {  
             class_replaceMethod(c,overrideSEL, method_getImplementation(origMethod), method_getTypeEncoding(origMethod));
        } else  {
             method_exchangeImplementations(origMethod,overrideMethod);
        }
    }

在本类中交换要考虑两种情况,第一种情况是要复写的方法并没有在目标类中实现,而是在其父类中实现了。第二种情况是这个方法已经存在于目标类中,对于第一种情况,应当先在目标类增加一个新的实现方法(override),然后将复写的方法替换为原先的实现,运行时函数class_addMethod 如果发现方法已经存在,会失败返回,如果添加成功(在父类中重写的方法),再把目标类中的方法替换为旧有的实现(译注:addMethod会让目标类的方法指向新的实现,使用replaceMethod再将新的方法指向原先的实现,这样就完成了交换操作。),如果添加失败了,就是第二情况(在目标类重写的方法)。这时可以通过method_exchangeImplementations来完成交换,对于第二种情况,因为class_getInstanceMethod 会返回父类的实现,如果直接替换,就会替换掉父类的实现,而不是目标类中的实现。

  1. 两个不同的类交换方法的实现。

     void hookInstanceMethod(Class origClass,SEL origSEL, Class newClass,SEL newSEL)
     {
         Method origMethod = nil;
         Method newMethod = nil;
    
         origMethod = class_getInstanceMethod(origClass, origSEL);
         newMethod = class_getInstanceMethod(newClass, newSEL);
    
         //先在原先的类添加新的Method
         if(class_addMethod(origClass, newSEL, method_getImplementation(origMethod), method_getTypeEncoding(origMethod))){
             method_exchangeImplementations(origMethod, newMethod);
         }
      }
    

对于第二种情况,在旧类中添加一个新方法名,这个新方法名称的实现指向旧方法的实现,这样在旧的类的旧方法的实现(imp)和新类的新方法实现(imp)条换之后,在调用旧方法的时候,这个实际就调用了新的实现,然后要在新的实现中还要调用旧的实现,这时候就是要调用旧类中我们添加的那个新方法名。

实战

上面写了那么多的理论是不是要来点“真枪实弹”的东西,不然扯那么多真的是站着说话腰不疼,那么现在,我拉着你,你跟着我进入一个场景:

在一个像往常一样的早上,你在来上班的路上望着晕沉沉的天空,貌似有种不好的预感要发生什么不好的事情,你到了办公室坐了下来喝了杯咖啡压压惊,这时候产品走了过来跟你讲:我昨天为我们几个流量产品申请了广点通的应用,但是因为有一两个产品流量量比较少,没申请下来,因为广点通的广告会根据Bundle ID来确认对应的应用,所以你那边技术能处理下这个Bundle ID,让不同应用使用同一个广点通的ID吗?好行,这个重任就交给你了。这时你想反驳点什么,但是还没开口话已经被产品说死了,这时你内心的想法:


hit.jpg

这时可能你准备抽出36米长的大刀, 此时产品突然抽出了贴身的长枪。


dou.gif

这时的你没有什么办法了,只能默默的低头按照产品的要求做事情,内心里的世界:


still.gif

开玩笑开玩笑,钱还是要赚的,然后你慢慢整理你扔掉的东西,打开电脑,留下了一个孤独无助的背影...

当然这时候你只要知道runtime 的swizzling,你就有思路了,不就是换个Bundle ID 嘛,是时候展现真正技术的时候了,这时候你打开ida ,将广点通的静态库拉入ida,没办法,你只能慢慢找,看是否有哪个类是专门来获取app信息的,你查看的每个类,F5,F5,F5...,当你点到GDTStatsMgr这个类的时候,这时候你惊奇的发现这个画面:

systemInfo.png

没错,找到类了,这个类就是获取一些app,手机系统信息的类,这时候你查看着每个方法,看哪个方法是获取Bundle ID的,这时候你发现了一个方法initializeData
initializeData.png

这个方法就是用来获取一些基础信息的,你看了下这个类的一些方法,好像没有关于Bundle ID的字段,这时候你发现前面那些方法图里面有个成员变量很奇怪:

-[GDTStatsMgr an]
-[GDTStats setAn:]

嗯,这时你在你的工程delegate里面添加上这句代码:

  hookInstanceMethod(NSClassFromString(@"GDTStatsMgr"),@selector(initializeData),[AppDelegate class],@selector(my_initializeData));

并写上要交换的方法:

 - (void)my_initializeData{
    [self my_initializeData];

    [self setValue:@"想要换成的Bundle ID" forKey:@"an"];
}

OK,这时你颤抖的左手按了下command + R, 这时候一个启动屏广告皓然展现在你的面前, 这时候兴奋的想要大叫一声,不过还是忍了下来,然后淡淡的走过去跟产品说: 你刚才的那个小需求我搞定了。然后转身走的时候留下程序员应有逼格的背影....

总结

OC 中Runtime的swizzling是一个很神奇的东西, 用的好的话可以解决各种疑难杂症,我们一般会在分类中+load写对应的交换, 想要知道为什么,可以点击这里Objective-C Method Swizzling 的最佳实践

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

推荐阅读更多精彩内容