废话不多说,分享两个简单的runtime使用场景:
1、在存储用户信息的时候使用的归档和解档;
2、替换系统方法。
归档和解档首先我们明确的知道需要实现两个协议方法:
解档:
- (instancetype)initWithCoder:(NSCoder *)decoder
归档:
- (void)encodeWithCoder:(NSCoder *)encoder
一般存储用户信息我们都是在model里面进行操作(因为这里是用来写用户属性的),通过KVC来进行属性的存取。
不使用runtime进行归档:
[encoder encodeObject:self.userName forKey:@"userName"];
如果有太多太多信息要存储这里我们就需要复制粘贴很多,所以这里使用runtime会方便很多(把类拿出去就可以直接使用)。
使用runtime进行归档( 导入头文件是必须的):
class_copyIvarList(<#__unsafe_unretained Class cls#>, <#unsigned int *outCount#>)
这里的IvarList可以理解为属性列表,后面的前一个参数为哪个类,第二个参数为属性个数无符号整形指针(这里我们要传一个地址)例如:
unsigned int count = 0;
Ivar *ivars = class_copyIvarList([AccountModel class], &count);
这里的返回值可以理解为一个指针数组(只是可以这么理解,其实不是,它就是一个Ivar类型)
这里我们获取到了属性个数,然后通过下标获取属性名,紧接着通过kvc进行归档
for (int i = 0; i < count; i++) {
// 根据下标获取得到Ivar
Ivar ivar = ivars[i];
// 通过ivar_getName获取属性名称(类型为char类型 c中的类型)
const char *name = ivar_getName(ivar);
// 将char类型通过utf8转换成oc中的string类型
NSString *key = [NSString stringWithUTF8String:name];
// 然后进行归档
[encoder encodeObject:[self valueForKey:key] forKey:key];
}
在arc机制下释放不了c 中copy的指针,所以这里我们需要手动释放一下
free(ivars);
解档:
解档其实和上面代码基本上就一样了
unsigned int count = 0;
Ivar *ivars = class_copyIvarList([AccountModel class], &count);
for (int i = 0; i < count; i++) {
Ivar ivar = ivars[i];
const char *name = ivar_getName(ivar);
NSString *key = [NSString stringWithUTF8String:name];
// 反归档
// 这里根据属性名称作为key值来获取value值
id value = [decoder decodeObjectForKey:key];
[self setValue:value forKey:key];
}
到这里归档解档就完成了,(代码直接复制到项目当中就可以使用的,不需要做任何修改)。
方法替换实例:
通常我们在发送请求的时候是这样的:
NSURL *url = [NSURL URLWithString:@"xxxxxxxxxx"];
NSURLRequest *request = [NSURLRequest requestWithURL:url];
如果连接中带有中文的时候这个请求就会失败:
NSURL *url = [NSURL URLWithString:@"xxxxxxxxxx/你好"];
当URL中存在汉字的时候,给个提示,但是现在我不想在这里做判断,因为假如说我程序中存在好多请求,我是不是每次遇到请求都要写着个判断来进行提示?所以我们来修改系统的URLWithString方法,给NSURL类做扩展,这里我们会发现,假如我们要重写
这个方法的话,他是个构造类方法,还是要用到URLWithString,所以这里不能重写它。
那么我们自己来构造一个方法
接下来我们要用runtime将我们自己的方法替换系统的方法,在替换之前,我们得知道我们在哪里替换,肯定是当程序一运行我们就编译(比viewdidload先)就是load方法
// 首先介绍下这两个方法:
// 这个是获取对象方法
class_getInstanceMethod(<#__unsafe_unretained Class cls#>, <#SEL name#>)
// 这个是获取类方法
class_getClassMethod(<#__unsafe_unretained Class cls#>, <#SEL name#>)
当然这里我们用到是肯定是类方法:
先获取系统的方法(返回类型是Method):
Method urlString = class_getClassMethod([NSURL class], @selector(URLWithString:));
在获取自己构造的方法(这里的类是NSURL,因为自己写的方法是对NSURL类做的扩展)
Method ylurlString = class_getClassMethod([NSURL class], @selector(YL_URLWithString:));
替换方法:
method_exchangeImplementations(urlString, ylurlString);
这个时候就完成了,controller中的代码一点都没有改变,还是用的URLWithString:
但是这时候运行,程序会crash,因为我们会发现,在自己写的构造方法中用到了URLWithString,当controller中运行URLWithString的时候,就会找到我们所构造的YLURLWithString,但是当走YLURLWithString里面代码的时候,又会走URLWithString,一直这样下去,造成了死循环,所以我们自己构造的方法当中应该将URLWithString改成YL_URLWithString
这样就不会有问题了。
简单的两个应用场景,哪里说的不详情或说错的地方请谅解!runtime其实挺有意思的,但是感觉挺难得(毕竟是底层的东西),以后会更深入的去学习一下。
最后推荐一本学习runtime的书:Objective-C 2.0运行时系统编程指南.pdf