有关Swizzling的一个问题

前言

最近在开发的时候遇到了一个Swizzling的问题,特别在此记录,希望有相同遭遇的朋友能参考。

问题

问题概况:被Swizzling的方法在调用原有实现的时候提示“Selector无法被响应”。
问题过程:项目中A框架使用Aspects(一个帮助开发者实现快速Swizzling的框架)来替换UIViewController- (void)viewDidLoad方法,这个步骤是先被执行的。之后的某个情况下,项目中B框架又使用自己写的Swizzling方法替换了UIViewController- (void)viewDidLoad方法。这时我发现,当代码运行到B框架的替换方法,并且要执行原有的方法时,会报Selector无法被响应的错。很奇怪,因为从B框架中Swizzling的实现上来看,会保存之前的IMP,所以应该能找到才对。

原因

于是我去看了Aspects的源码,发现它的实现是这样的(假如现在要Swizzling classA类的methodA方法):

  1. 在methodA被Swizzling的时候先看classA的-forwardInvocation:方法是否被替换,如果不是,就使用自己的-AspectsForwardInvocation:方法来替换classA的-forwardInvocation:
  1. 将methodA的IMP指向_objc_msgForward,这是一个全局 IMP,OC 调用方法不存在时都会转发到这个 IMP 上,这样做了之后,当methodA被调用的时候,就会先进入-AspectsForwardInvocation:方法。
  2. 接下来我们看-AspectsForwardInvocation:的实现,这个方法会将NSInvocation的Selector拿出来,再拼上一个前缀(aspects_),然后检查这个方法是否是已经被Aspects替换过,如果是,就查询相关的实现并执行,如果不是,就执行原有的-forwardInvocation:方法来进行转发。

这下问题就很清楚了,当Aspects的Swizzling方法先被执行的时候,原方法Selector对应的IMP已经指向_objc_msgForward,所以当另一个框架再进行Swizzling的时候,存起来的原有实现就是这个“错的”_objc_msgForward。那么当执行完自己加的代码后,想要再通过objc_msgSend执行原有实现,就是会找不到,因为原有实现已经被替换为_objc_msgForward,而真的IMP由于被Aspects先Swizzling掉了,所以找不到!

具体Swizzling结果如下图:

Aspects处理结果(先)
另一处理结果(后)

具体执行逻辑如下图:

执行流程

解决办法

依照上面的说法,当项目里面有类似Aspects这种Swizzling方式,而且它先于其他Swizzling方式执行,且Swizzling了相同的代码,那么这个问题就会发生了,目前知道的另一个这么做的库是JSPatch(具体可以参考它Wiki中方法替换章节的第5小节)。使用Aspects库的人数我无法估计,但JSPatch的应该不少吧。所以给出合理的解决办法还是很有必要的。

方案1:如果在你的Swizzling方法内部需要调用原有方法,那么在执行原有方法的IMP之前先判断一下,如果为_objc_msgForward(或_objc_msgForward_stret),那么就使用原有Selector拼成一个NSInvocation,再使用objc_msgSend执行。使用原有Selector的原因是:在进入已被替换的-forwardInvocation:时,只有原有Selector可以帮助找到真实的实现。使用NSInvocation进行消息转发的原因是:这样可以走到-forwardInvocation:方法。这种方法的缺点是需要在每个地方都做处理,而且之前的判断也比较特殊。

方案2:参考Aspects或JSPatch相关代码,将-forwardInvocation:也进行Swizzling,在自己的-forwardInvocation:方法中进行同样的操作,就是判断传入的NSInvocation的Selector,如果是自己可以识别的Selector,那么就将Selector变为原有Selector在执行,如果不识别,就直接转发。(这也是为什么我们项目中Aspects和JSPatch不冲突的原因,因为都将被Swizzling的方法指向了_objc_msgForward(或_objc_msgForward_stret),然后再监控-forwardInvocation:方法,使得方法最终通过原有Selector经消息转发流程得到正确的实现)。

最后

欢迎讨论,欢迎指出问题。

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

推荐阅读更多精彩内容