美文网首页
IOS开发 — 之 RunTime

IOS开发 — 之 RunTime

作者: 余立徽 | 来源:发表于2017-11-02 16:49 被阅读0次

    学习IOS很久了,还没有怎么去了解runTime。今天去简单的去了解了一下runTIme。

    1、Obj 方法的底层实现是发消息

    图1

    首先我们新建一个NSObject 类 Person,然后去 Controller 中创建,上图中 在[Person alloc] 的时候,在内存中给 Person 分配一个内存空间,然后 init 初始化。

    在底层的实现相当于发送消息类,在 #import <objc/message.h > 中相当于:

    图2

    相当于是  objc_msgSend() 对应传参为 id 方法类型 ,和需要实现的方法。第一个参数 是方法的编号, 第二个参数是 去找到方法的实现。也可以直接用KVO的形式。也可以这么写成:

    图3

    2、利用 runTime 去修改系统方法

    NSURL 我们很常用的一个方法。我们在  NSURL 中 URLWithString: 方法用到的都很多,但是这个方法中存在一个坑,也就是URLWithString; 后面接的参数中不能包含有中文,当包含有中文的情况下,url的值为nil了,在项目中都已经用到很多的地方,每个地方每个地方的去修改显的就很麻烦了,但是我们可以用 runTime 去替换掉上面的 URLWithString:方法。

    首先我们新建一个NSURL 扩展, 创建一个要替换的方法:

    图4

    注意:上图 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];)再次运行的时候就不会为至为空了。这样我们就实现了,方法的替换。

    接下来我们验证一下:

    图5 图6

    上面运行时也就很容易的看到了,系统的方法被我们替换了。

    3、runTime解决调用了一个未实现的方法引起的崩溃

    在Person类中,有一个方法可以 捕获到 未被实现的方法。

    +(BOOL)resolveInstanceMethod:(SEL)sel;

    图7

    这是在 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  表示 整型】等等。

    当前是为传参数。如果需要传参数的话的主意了,如下图所示:

    图8

    在 newMethod 方法中我们给了三个参数,其中前面的两个参数是在未传参数时候 是省略掉的。真正的第三个参数才是我们传的参数。如果只写:

    void newMethod(NSString *obc) { } 

    那就是不对的了,我们打印出 obc  参数就不是我们传递的参数,第一个参数为当前的类,应该指的是 Person,所以当传参数的时候 应该补全前面两个参数。

    接下来我们来看一下结果:

    图9 图10

    从上面可以看出 那边传递过来的参数正常的传递过来了。这样就不会导致程序的崩溃了,接下来我们看一下,如果我们在 newMethod 中只写一个参数看看:

    图11

    可以看出如果我们只写一个参数 那就默认是 当前的类,这里是一个坑,传第参数的时候的注意了。这里我们举例的『实例方法』的检测,里面也有检测『类方法』的,这个留给大家去完成吧。

    图12

    4、Ivar 在IOS开发中的强大之处

    1、Ivar 去获取到类的属性,既可以获取到 .h 和 .m 文件的属性都可以。

    图12

    可以看到获取到的熟悉有3个,然后我们看下 Person 文件。发现 .h 和 .m 文件中都获取到属性:

    图13  .m 文件 图14 .h 文件

    我们现在添加两个私有变量,看看效果。

    图14

    很明显也是可以扑捉到的。

    图15

    但是我们发现,私有变量获取的没有下划线,而全局变量前面带有下划线。这样我们就可以去查看到类里面那些属于私有变量。

    runTime的实际应用

    1、动态给分类添加属性

    这个应该使用的比较频繁,通过runtime动态添加属性,可以给系统类添加自定义属性,灵活使用,可以带来神奇的效果。

    2、方法的交换swizzling

    method_exchangeImplementations

    3、字典转模型

    4、获取所有的私有属性和方法

    这个在判断是否子类重写了父类的方法时会用到。

    Ivar*ivars=class_copyIvarList([UIPageControl class],&count);

    5、对私有属性修改

    6、归档:解档

    7、动态的添加方法

    相关文章

      网友评论

          本文标题:IOS开发 — 之 RunTime

          本文链接:https://www.haomeiwen.com/subject/ngxapxtx.html