runtime的理解
今天同学问我什么是runtime,我想了想就简单总结出了这篇文章,以下仅供参考.
我所理解的runtime是一个使用C编写的库,为C添加了面向对象的特性,它是一个库(Runtime Library中文:运行时库).在这个库中可以用C函数来实现方法,对象也可以用C语言的结构体来表示...所有oc的方法的背后都是通过runtime来运行的.
如图创建一个Person
类
通过终端编译一下.m :
1.终端切换到工程的目录下
2.然后输入命令
clang -rewrite-objc main.m
3.找到编译后的main.cpp如下图
然后我点开cpp文件(C++文件)发现一大推代码,在最后发现了如下图的代码:
结论:
Person *p = [[Person alloc] init]
最终被编译成了下面的样子(也就是上图中的样子)
Person *p = ((Person *(*)(id, SEL))(void *)objc_msgSend)((id)((Person *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("Person"), sel_registerName("alloc")), sel_registerName("init"));
我发现调用方法其实最终被编译成发送具体某一个消息.
runtime如何应用,怎么用?
先说用处:
1.拦截系统自带的方法调用(好多人用这种方法省去了很多重复性的操作,比如拦截imageName方法,不同系统版本环境下对图片做不同操作)
2.给分类增加个属性(我的渐变色背景二那篇文章里面的代码有用到一些)
3.归档解档
一、拦截系统自带的方法,调用自己的方法
导入<objc/runtime.h>
需要用到
//获取类方法
Method class_getClassMethod(Class cls , SEL name)
//获取对象方法
Method class_getInstanceMethod(Class cls , SEL name)
//交换方法
void method_exchangeImplementations(Method m1 , Method m2)
具体场景
需求:iOS7 后需要版本适配,不同系统版本使用不同样式图片(拟物化和扁平化),如何通过不去手动一个个修改每个UIImage的imageNamed:方法就可以实现为该方法中加入版本判断语句?
步骤:
1.创建UIImage分类
2.m里自定义一个方法
3.在load里做交换方法(load只执行一次)
4.如下图;
实现之后,我在VC里调用下imageNamed方法如下:
通过上面的实现就拦截到了系统方法针对不同系统版本做我想做的操作.
二、给分类设置属性
代码可以参考渐变色背景(二)那篇文章,这里我对方法总结一下
//set方法里实现
//id object 哪一个对象设置属性
//const void *key 通过key取出存的value值
//value 给属性设置的值(存的value值)
//policy:
//policy设置的值和assgin copy retain差不多,只不过变成:
//`OBJC_ASSOCIATION_ASSIGN`,
//`OBJC_ASSOCIATION_COPY_NONATOMIC`,
//`OBJC_ASSOCIATION_RETAIN_NONATOMIC`
void objc_setAssociatedObject(id object , const void *key ,id value ,objc_AssociationPolicy policy)
//get方法里实现,对应的参数说明看set
id objc_getAssociatedObject(id object , const void *key)
三、NSCode协议下,归档解档
//获取所有成员变量和变量总数
Ivar *class_copyIvarList(Class cls , unsigned int *outCount)
//获得成员变量的名字
const char *ivar_getName(Ivar v)
//获得成员变量的类型
const char *ivar_getTypeEndcoding(Ivar v)
//归档
- (void)encodeWithCoder:(NSCoder *)coder
{
//属性的个数
unsigned int count = 0;
//
Ivar * ivars = class_copyIvarList([self class], &count);
for (int i = 0; i < count; i++) {
//取出对应的Ivar
Ivar ivar = ivars[i];
//拿到名称
const char * name = ivar_getName(ivar);
//OC 字符串
NSString * key = [NSString stringWithUTF8String:name];
//归档
[coder encodeObject:[self valueForKey:key] forKey:key];
}
free(ivars);
}
//解档
- (instancetype)initWithCoder:(NSCoder *)coder
{
if (self = [super init]) {
//属性的个数
unsigned int count = 0;
Ivar * ivars = class_copyIvarList([self class], &count);
for (int i = 0; i < count; i++) {
//取出对应的Ivar
Ivar ivar = ivars[i];
//拿到名称
const char * name = ivar_getName(ivar);
//OC 字符串
NSString * key = [NSString stringWithUTF8String:name];
//解档
id value = [coder decodeObjectForKey:key];
//KVC--设置值到属性上面!!!
[self setValue:value forKey:key];
}
free(ivars);
}
return self;
}
我日后再深入了解,及时更新文章,thank you