编码篇-低耦合代码注入

前言

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


一个问题:

`如何在一个大的项目中使所有的 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。**

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Spark学习技巧

Flink DataSet编程指南-demo演示及注意事项

Flink中的DataStream程序是对数据流进行转换的常规程序(例如,过滤,更新状态,定义窗口,聚合)。数据流的最初的源可以从各种来源(例如,消息队列,套接...

3.9K12
来自专栏difcareer的技术笔记

JNI实现源码分析【四 函数调用】正文0x01:dvmCallMethodV0x02:nativeFunc0x03: 何时赋值

有了前面的铺垫,终于可以说说虚拟机是如何调用JNI方法的了。JNI方法,对应Java中的native方法,所以我们跟踪对Native方法的处理即可。

984
来自专栏向治洪

ES7和ES8新特性介绍

概述 JavaScript,作为一门处于高速发展期的开发语言,正在变的越来越完善、稳定。我们必须拥抱这些变化,并且我们需要把ES8加入到我们的技术栈中。 ECM...

6196
来自专栏菜鸟致敬

记一次两小时的js编程学习

1.弱类型语言 2.解释型语言 3.客户端语言 对于有学习Java、C以及Python一类的人来说,最熟悉的莫过于这些都是强类型语言。它们严格的遵守自身的规定,...

852
来自专栏逍遥剑客的游戏开发

C++的反射和序列化

1532
来自专栏PHP在线

正则表达式中的子组模式

作者:西瓜玩偶(racnil070512 at hotmail dot com) 一、基础知识 在PCRE正则表达式中,我们可以利用圆括号定义一个子组,我们...

39712
来自专栏java一日一条

最全面的 Android 编码规范指南

这份文档参考了 Google Java 编程风格规范和 Google 官方 Android 编码风格规范。该文档仅供参考,只要形成一个统一的风格,见量知其意就可...

1604
来自专栏Golang语言社区

动手实现一个JSON验证器(上)

分析 既然要验证JSON的有效性,那么必然需要清楚的知道JSON格式,这个在JSON官网已经给我们画出来了: ? ? ? ? ? 从官方的图上面可以看出,JSO...

5237
来自专栏向治洪

ES7、ES8新特性

概述 JavaScript,作为一门处于高速发展期的开发语言,正在变的越来越完善、稳定。我们必须拥抱这些变化,并且我们需要把ES8加入到我们的技术栈中。 E...

1.4K5
来自专栏IMWeb前端团队

当JavaScript遇上UINT64

导语:写下这篇文章的缘由是因为在项目过程中,碰到了一个使用JavaScript处理 UINT64 类型数字的坑。 与大部分现代编程语言(包括几乎所有的脚本语言...

3000

扫码关注云+社区

领取腾讯云代金券