Runtime 在项目中的用处.
2017-11-18 本文已影响0人
Damoness
1. 全局替换系统的方法为自己的方法
使用UIImage imageNamed 加载图片时,由于需要手动输入名称,这个很容易导致加载的图片对象为空, 这个问题很容易忽视. 这时如果替换掉系统的方法, 并判断是否为空,并打印出来 这就很直观.
@implementation UIImage (image)
//分类被加入运行时的时候就调用,在main方法之前.
+(void)load{
// 1.获取系统方法
Method imageNamedMethod = class_getClassMethod(self, @selector(imageNamed:));
// 2.自己实现的方法
Method ln_imageNamedMethod = class_getClassMethod(self, @selector(ln_imageNamed:));
//3.交换方法
method_exchangeImplementations(imageNamedMethod, ln_imageNamedMethod);
}
+(UIImage *)ln_imageNamed:(NSString *)name{
//这里是关键,因为ln_imageNamed已经 和 imageNamed交换,这里实际调用的是imageNamed,并不会导致循环调用.
UIImage *image = [UIImage ln_imageNamed:name];
if (image) {
NSLog(@"runtime添加额外功能--加载成功");
}else{
NSLog(@"runtime添加额外的功能--加载失败");
}
return image;
}
@end
//使用
[UIImage imageNamed:@"imageName"];
2. 给系统的类增加属性.
系统的类不能满足自己的需求,需要增加额外的属性.其实本质就是给这个类添加关联,并不是直接把这个值的内存空间添加到类存空间。
@interface NSObject (Property)
@property NSString *name;
@end
@implementation NSObject (Property)
- (void)setName:(NSString *)name{
//将属性的值和传递进的name关联
objc_setAssociatedObject(self, @"name", name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
-(NSString *)name{
//获取关联属性的值
return objc_getAssociatedObject(self, @"name");
}
@end
3. 自己实现字典转属性
@interface NSObject (DictToModel)
+(instancetype)modelWithDict:(NSDictionary *)dict;
@end
@implementation NSObject (DictToModel)
+(instancetype)modelWithDict:(NSDictionary *)dict{
//1.创建对象
id objc = [[self alloc]init];
// 2.利用runtime给对象中的属性赋值
/**
class_copyIvarList: 获取类中的所有成员变量
Ivar:成员变量
第一个参数:表示获取哪个类中的成员变量
第二个参数:表示这个类有多少成员变量,传入一个Int变量地址,会自动给这个变量赋值
返回值Ivar *:指的是一个ivar数组,会把所有成员属性放在一个数组中,通过返回的数组就能全部获取到。
count: 成员变量个数
*/
unsigned int count = 0;
//3.获取类中所有的成员变量
Ivar *ivarList = class_copyIvarList(self, &count);
//遍历数组所有成员变量
for (int i = 0; i < count; i++) {
//从数组中取出对应的成员
Ivar ivar = ivarList[i];
//获取成员变量的名字
NSString *ivarName = [NSString stringWithUTF8String:ivar_getName(ivar)];
//处理成员变量 -> 字典中的key(去掉_ )
NSString *key = [ivarName substringFromIndex:1];
NSLog(@"ivarName:%@",key);
//根据成员属性名去字典查找对应的value
id value = dict[ivarName];
//找到value
if (value) {
//模型中的属性赋值
[objc setValue:value forKey:key];
}
}
return objc;
}
@end
4.动态添加方法
如果一个类方法非常多,加载类到内存的时候比较耗费资源,需要给每个方法生产映射表,可以动态给某个类,添加方法来解决.
@implementation PersonModel
// 没有返回值,1个参数
// void,(id,SEL)
void aaa(id self, SEL _cmd, NSNumber *meter) {
NSLog(@"跑了%@米", meter);
}
// 任何方法默认都有两个隐式参数,self,_cmd(当前方法的方法编号)
// 什么时候调用:只要一个对象调用了一个未实现的方法就会调用这个方法,进行处理
// 作用:动态添加方法,处理未实现
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
// [NSStringFromSelector(sel) isEqualToString:@"run"];
if (sel == NSSelectorFromString(@"run:")) {
// 动态添加run方法
// class: 给哪个类添加方法
// SEL: 添加哪个方法,即添加方法的方法编号
// IMP: 方法实现 => 函数 => 函数入口 => 函数名(添加方法的函数实现(函数地址))
// type: 方法类型,(返回值+参数类型) v:void @:对象->self :表示SEL->_cmd
class_addMethod(self, sel, (IMP)aaa, "v@:@");
return YES;
}
return [super resolveInstanceMethod:sel];
}
@end
//使用
// 默认PersonModel,没有实现run:方法,可以通过performSelector调用,但是会报错。
// 动态添加方法就不会报错
PersonModel *person = [PersonModel new];
[person performSelector:@selector(run:) withObject:@10];
//输出
跑了10米