学习IOS很久了,还没有怎么去了解runTime。今天去简单的去了解了一下runTIme。
1、Obj 方法的底层实现是发消息
首先我们新建一个NSObject 类 Person,然后去 Controller 中创建,上图中 在[Person alloc] 的时候,在内存中给 Person 分配一个内存空间,然后 init 初始化。
在底层的实现相当于发送消息类,在 #import <objc/message.h > 中相当于:
相当于是 objc_msgSend() 对应传参为 id 方法类型 ,和需要实现的方法。第一个参数 是方法的编号, 第二个参数是 去找到方法的实现。也可以直接用KVO的形式。也可以这么写成:
2、利用 runTime 去修改系统方法
NSURL 我们很常用的一个方法。我们在 NSURL 中 URLWithString: 方法用到的都很多,但是这个方法中存在一个坑,也就是URLWithString; 后面接的参数中不能包含有中文,当包含有中文的情况下,url的值为nil了,在项目中都已经用到很多的地方,每个地方每个地方的去修改显的就很麻烦了,但是我们可以用 runTime 去替换掉上面的 URLWithString:方法。
首先我们新建一个NSURL 扩展, 创建一个要替换的方法:
注意:上图 NSURL *url = [NSURL HK_urlString:urlString]; 不能使用 NSURL *url = [NSURL URLWithString:urlString]; 否则是死循环了。
我们现在要做的就是 用我们写好的 HK_urlString:方法去替换 系统的 URLWithString:方法。那我们接下来就 用runTime去修改,思路就是:我们在系统运行到 URLWithString:这个方法的时候 去拿到这个方法,然后我们去修改。于是我们要在项目运行的最开始的位置去 使用这个方法。于是我们想到了 load 方法, load 方法会在项目运行前的加载的过程中 就会被读入到内存之中。在上面自定义的 HK_urlString:方法中,我们可以对参数 urlString 进行编码,使用 NSUTF8StringEncoding 编码(urlString = [urlString stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];)再次运行的时候就不会为至为空了。这样我们就实现了,方法的替换。
接下来我们验证一下:
上面运行时也就很容易的看到了,系统的方法被我们替换了。
3、runTime解决调用了一个未实现的方法引起的崩溃
在Person类中,有一个方法可以 捕获到 未被实现的方法。
+(BOOL)resolveInstanceMethod:(SEL)sel;
这是在 resolveInstanceMethod 方法中去捕获到未实现的方法,然后我们自己去 自动的去新建一个实现的方法(newMethod)。
这里我来简单的介绍一下 class_addMethod 方法的参数:
class_addMethod(<#Class _Nullable __unsafe_unretained cls#>, <#SEL _Nonnull name#>, <#IMP _Nonnull imp#>, <#const char * _Nullable types#>)
前面的三个参数也就不做介绍了,截图中都有说明,我们来说下第四个参数 types,表示 IMP 去实现方法的参数类型。上图中的第四个参数可以传: "v@" ,而下面的 『图8』的第四个参数可以传:"v@:@" 。其中:【 v 表示 void】、【@ 表示 oc类型参数】、【: 表示 传参】、【i 表示 整型】等等。
当前是为传参数。如果需要传参数的话的主意了,如下图所示:
在 newMethod 方法中我们给了三个参数,其中前面的两个参数是在未传参数时候 是省略掉的。真正的第三个参数才是我们传的参数。如果只写:
void newMethod(NSString *obc) { }
那就是不对的了,我们打印出 obc 参数就不是我们传递的参数,第一个参数为当前的类,应该指的是 Person,所以当传参数的时候 应该补全前面两个参数。
接下来我们来看一下结果:
从上面可以看出 那边传递过来的参数正常的传递过来了。这样就不会导致程序的崩溃了,接下来我们看一下,如果我们在 newMethod 中只写一个参数看看:
可以看出如果我们只写一个参数 那就默认是 当前的类,这里是一个坑,传第参数的时候的注意了。这里我们举例的『实例方法』的检测,里面也有检测『类方法』的,这个留给大家去完成吧。
4、Ivar 在IOS开发中的强大之处
1、Ivar 去获取到类的属性,既可以获取到 .h 和 .m 文件的属性都可以。
可以看到获取到的熟悉有3个,然后我们看下 Person 文件。发现 .h 和 .m 文件中都获取到属性:
我们现在添加两个私有变量,看看效果。
很明显也是可以扑捉到的。
但是我们发现,私有变量获取的没有下划线,而全局变量前面带有下划线。这样我们就可以去查看到类里面那些属于私有变量。
runTime的实际应用
1、动态给分类添加属性
这个应该使用的比较频繁,通过runtime动态添加属性,可以给系统类添加自定义属性,灵活使用,可以带来神奇的效果。
2、方法的交换swizzling
method_exchangeImplementations
3、字典转模型
4、获取所有的私有属性和方法
这个在判断是否子类重写了父类的方法时会用到。
Ivar*ivars=class_copyIvarList([UIPageControl class],&count);