前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >编码篇-低耦合代码注入

编码篇-低耦合代码注入

作者头像
進无尽
发布2018-09-12 18:33:48
5760
发布2018-09-12 18:33:48
举报
文章被收录于专栏:進无尽的文章進无尽的文章

前言

我下面要将的内容也许网上已经有很多相关的介绍了,但是我还是会写出这篇文章,一来是对自己学习的总结,虽然总结的有些晚,如果你仔细看,会发现我的文章有别处没有的内容介绍,而且都是亲测过的。


一个问题:

`如何在一个大的项目中使所有的 VC 都在试图将要出现的时候打印出当前类的名称,而且要不影响到原有方法的执行?

方案

  • 使用 继承,在父类的 viewWillAppear 中写入相关的代码即可,如果是新项目自然是可以的。
  • 使用代码注入 就是传说中的 Runtime - Method Swizzling。方法交换。

思考

  • 我们不希望改变原有类的对应方法,如果在Catagory (非系统级别的才可以重写,无法通过类别重写系统级别的类方法) 中重写一个方法,就会覆盖它的原有方法实现,但是,这样做以后就没有办法调用系统原有的方法,但是在类别中重写系统方法会有警告,并且在出问题时不容易排查。
  • 在 Objective-C 的运行时中,每个类有两个方法都会自动调用。+load 是在一个类被初始装载时调用,+initialize 是在应用第一次调用该类的类方法或实例方法前调用的。两个方法都是可选的,并且只有在方法被实现的情况下才会被调用。
  • +load,对于加入运行期系统中的每个类(class)及分类(category)来说,必定会调用此方法,如果分类和其所属的类都定义了 +load方法,则先调用子类里的+load方法,最后再调用类别(分类)中的+load方法。(在类别中定义的+load发法,有多少个类就会被调用多少次,网上有人说只会调用一次是错误的,亲测)。
  • 由于 swizzling 改变了全局的状态,dispatch_once能保证在不同的线程中也能确保代码只执行一次。 + (void)load { //在debug模式下进行方法的交换 #ifdef DEBUG static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ //原本的viewWillAppear方法 Method viewWillAppear11= class_getInstanceMethod([self class], @selector(viewWillAppear:)); //需要替换成 能够输出日志的viewWillAppear Method logViewWillAppear11 = class_getInstanceMethod([self class], @selector(logViewWillAppear:)); BOOL addSucc = class_addMethod([self class], @selector(viewWillAppear:), method_getImplementation(logViewWillAppear11), method_getTypeEncoding(logViewWillAppear11)); if (addSucc) { class_replaceMethod([self class], @selector(logViewWillAppear:), method_getImplementation(viewWillAppear11), method_getTypeEncoding(viewWillAppear11)); } else{ method_exchangeImplementations(viewWillAppear11, logViewWillAppear11); } }); #endif } - (void)logViewWillAppear:(BOOL)animated { NSString *className = NSStringFromClass([self class]); //在这里,你可以进行过滤操作,指定哪些viewController需要打印,哪些不需要打印 if ([className hasPrefix:@"UI"] == NO) { NSLog(@"%@ will appear OOOOOO",className); } #下面方法的调用,其实是调用viewWillAppear // [self logViewWillAppear:animated]; }`

这样我们的目的就实现了。

使用class_getClassMethod 一直失败,知道原因后会更新使用这个方法的示例。 这里解释下上面的方法,它的目的是为了使用一个重写的方法替换掉原来的方法。但被重写的方法可能是在父类中重写的,也可能是在子类中重写的。 对于第一种情况,应当先在目标类增加一个新的实现方法,class_addMethod:如果发现方法已经存在,会失败返回,如果返回成功:则说明被替换方法没有存在,我们需要先把这个方法实现,然后再用我们自定义的方法去替换被替换的方法。这样的逻辑判断是比价安全的,因为消息转发机制的存在,当前类没有系统方法的实现即系统方法实现在父类时,class_getInstanceMethod会返回父类的实现,如果直接调用method_exchangeImplementations,则会替换掉父类的实现,而不是当前类的实现,则不能达到预期的效果。

** 注意:要说明一下,上述方法实现了方法的拦截和替换,但是因为是在类别中实现的所以替换的是UIViewController中的方法,而很多其它 VC都是继承自 UIViewController,因为 [Super viewWillAppear ]的存在你会发现,其它的VC中还是会执行它自己viewWillAppear 的类容,因为你拦截并换的只是它父类中的viewWillAppear而不是它本身的viewWillAppear。**

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2016.11.10 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 前言
  • 一个问题:
  • 方案
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档