runtime是面试当中经常会问到的一个问题,可是这个在平常当中使用确实不多,至少本人是这样,以前一直只知道有那么个东西,一个没尝试过,现在来看看。
runtime的优点:大家都知道OC是一个动态语言,那runtime就是实现动态的一个方式,可以动态生成类,添加方法和属性。动态生成的好处:
- 可以优化内存,类似于懒加载那样,在需要的时候加载到内存
- 交换方法实现(交换系统的方法)
- 遍历类的所有成员变量(修改textfield的占位文字颜色、字典转模型、自动归档解档)
- 利用关联对象(AssociatedObject)给分类添加属性
- 可用统计和处理传错方法或未实现的方法
动态添加方法:
如果是正常的方法调用,OC中应该是生成一个对象a,调用test方法:[a test],但是如果对象a没有test方法,这样写编译就会报错。现在我们新建一个Dynamic类,继承NSObject,现在我们需要Dynamic的一个对象运行test方法,但是我们不需要任何的声明,在调用的时候,我们需要用到performSelector方法:
Dynamic *dynamic = [[Dynamic alloc] init];
[dynamic performSelector:@selector(test)];
这样写的话会有一个警告,说明你的类中并没有test方法,在Dynamic.h的方法中不需要添加任何东西,Dynamic.m中如下:
#import "Dynamic.h"
#import <objc/runtime.h>
@implementation Dynamic
+ (BOOL)resolveInstanceMethod:(SEL)sel {
if (sel = @selector(test)) {
class_addMethod([self class], sel, class_getMethodImplementation(self, @selector(implement)), "v@:");
return YES;
}
return [super resolveInstanceMethod:sel];
}
- (void)implement {
NSLog(@"动态添加了一个方法");
}
这样就可以动态完成了对于Dynamic类的test方法的动态添加,下面我会详细讲一讲里面的内容。
当我们用到 [dynamic performSelector:@selector(test)]时,系统由于找不到test方法,会调用Dynamic类的
+ (BOOL)resolveInstanceMethod:(SEL)sel ;
这个方法是当类对象调用方法时用到的,还有一个比较相像的,+ (BOOL)resolveClassMethod:(SEL)sel;这个是调用类方法时会进入的,可以动态添加类方法。
这里注意下调用顺序,当方法有实现时,是先进入方法的实现里,然后进入resolveInstanceMethod方法,对象执行任何方法最后都会进入resolveInstanceMethod方法。
下面就是class_addMethod(Class _Nullable __unsafe_unretained cls, SEL _Nonnull name, IMP _Nonnull imp, const char * _Nullable types),这个就是动态添加的runtime中方法,下面讲解4个参数:
* @param cls The class you want to modify.
* @param name A selector that identifies the method whose implementation you want to replace.
* @param imp The new implementation for the method identified by name for the class identified by cls.
* @param types An array of characters that describe the types of the arguments to the method.
这是苹果对于参数的解释
1、Class _Nullable __unsafe_unretained cls 类对象,给哪个类添加方法
2、SEL _Nonnull name 方法名,哪个方法名是你想去替换实现的
3、IMP _Nonnull imp 替换的实现,注意这里IMP是方法的指针
4、const char * _Nullable types 参数类型,这里的有一种特殊的表达方式,无返回值不带参数的为"v@:",v代表void,@代表对象,:代表方法,但是你看我上面的代码,并没有一个参数,这是因为每一个方法会默认隐藏两个参数,self、_cmd,self代表方法调用者,_cmd代表这个方法的SEL,其他的缩写看下面的链接。
然后就是写下面的方法实现了,就完成了。对于方法的实现,苹果文档中是用C语言写的,不知道为什么,可能用C有更好的效率,也希望有知道的人留言。C语言实现方法的,第三个参数应该 (IMP) implement 这样书写。
上面的实现是无参数的,当你需要传入参数时,有performSelector: withObject:这个方法,最多可以传2个参数,传三个以上参数也有方法,自行百度。
多传参数后,方法的实现也需要输入对应的参数,class_addMethod的第四个参数也需要添加相应个数的@,不然的话会崩溃。动态添加方法基本就是这些了。