iOS开发中runtime常用的几种方法示例总结

合集下载
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

iOS开发中runtime常⽤的⼏种⽅法⽰例总结
前⾔
Objective-C runtime是⼀个实现Objective-C语⾔的C库。

它是⼀门编译型语⾔、也是⼀门动态型的语⾔(这⾥强调下OC是静态类型语⾔),之前没接触runtime的时候也不觉着它有多重要,接触之后才发现其实runtime挺强⼤的。

就拿我们在iOS开发中所使⽤的OC编程语⾔来讲,OC之所以能够做到即是编译型语⾔,⼜能做到动态语⾔,就是得益于runtime的机制。

最近公司项⽬中⽤了⼀些 runtime 相关的知识, 初看时有些蒙, 虽然⽤的并不多, 但还是想着系统的把 runtime 相关的常⽤⽅法整理⼀下, ⾃⼰以后⽤着⽅便, 也希望对看到的朋友有所帮助.
⼀、runtime 简介
runtime 简称运⾏时,是系统在运⾏的时候的⼀些机制,其中最主要的是消息机制。

它是⼀套⽐较底层的纯 C 语⾔ API, 属于⼀个 C 语⾔库,包含了很多底层的 C 语⾔ API。

我们平时编写的 OC 代码,在程序运⾏过程时,其实最终都是转成了 runtime 的 C 语⾔代码。

如下所⽰:
// OC代码:
[Person coding];
//运⾏时 runtime 会将它转化成 C 语⾔的代码:
objc_msgSend(Person, @selector(coding));
⼆、相关函数
// 遍历某个类所有的成员变量
class_copyIvarList
// 遍历某个类所有的⽅法
class_copyMethodList
// 获取指定名称的成员变量
class_getInstanceVariable
// 获取成员变量名
ivar_getName
// 获取成员变量类型编码
ivar_getTypeEncoding
// 获取某个对象成员变量的值
object_getIvar
// 设置某个对象成员变量的值
object_setIvar
// 给对象发送消息
objc_msgSend
三、相关应⽤
更改属性值
动态添加属性
动态添加⽅法
交换⽅法的实现
拦截并替换⽅法
在⽅法上增加额外功能
归档解档
字典转模型
以上⼋种⽤法⽤代码都实现了, ⽂末会贴出代码地址.
runtime
四、代码实现
要使⽤runtime,要先引⼊头⽂件#import <objc/runtime.h>
4.1 更改属性值
⽤ runtime 修改⼀个对象的属性值
unsigned int count = 0;
// 动态获取类中的所有属性(包括私有)
Ivar *ivar = class_copyIvarList(_person.class, &count);
// 遍历属性找到对应字段
for (int i = 0; i < count; i ++) {
Ivar tempIvar = ivar[i];
const char *varChar = ivar_getName(tempIvar);
NSString *varString = [NSString stringWithUTF8String:varChar];
if ([varString isEqualToString:@"_name"]) {
// 修改对应的字段值
object_setIvar(_person, tempIvar, @"更改属性值成功");
break;
}
}
4.2 动态添加属性
⽤ runtime 为⼀个类添加属性, iOS 分类⾥⼀般会这样⽤, 我们建⽴⼀个分类, NSObject+NNAddAttribute.h, 并添加以下代码:
- (void)setName:(NSString *)name {
objc_setAssociatedObject(self, @"name", name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (NSString *)name {
return objc_getAssociatedObject(self, @"name");
}
这样只要引⽤ NSObject+NNAddAttribute.h, ⽤ NSObject 创建的对象就会有⼀个 name 属性, 我们可以直接这样写:
NSObject *person = [NSObject new];
= @"以梦为马";
4.3 动态添加⽅法
person 类中没有 coding ⽅法,我们⽤ runtime 给 person 类添加了⼀个名字叫 coding 的⽅法,最终再调⽤coding⽅法做出相应. 下⾯代码的⼏个参数需要注意⼀下:
- (void)buttonClick:(UIButton *)sender {
/*
动态添加 coding ⽅法
(IMP)codingOC 意思是 codingOC 的地址指针;
"v@:" 意思是,v 代表⽆返回值 void,如果是 i 则代表 int;@代表 id sel; : 代表 SEL _cmd;
“v@:@@” 意思是,两个参数的没有返回值。

*/
class_addMethod([_person class], @selector(coding), (IMP)codingOC, "v@:");
// 调⽤ coding ⽅法响应事件
if ([_person respondsToSelector:@selector(coding)]) {
[_person performSelector:@selector(coding)];
self.testLabelText = @"添加⽅法成功";
} else {
self.testLabelText = @"添加⽅法失败";
}
}
// 编写 codingOC 的实现
void codingOC(id self,SEL _cmd) {
NSLog(@"添加⽅法成功");
}
4.4 交换⽅法的实现
某个类有两个⽅法, ⽐如 person 类有两个⽅法, coding ⽅法与 eating ⽅法, 我们⽤ runtime 交换⼀下这两个⽅法, 就会出现这样的情况, 当我们调⽤ coding 的时候, 执⾏的是 eating, 当我们调⽤ eating 的时候, 执⾏的是 coding, 如下⾯的动态效果图.
Method oriMethod = class_getInstanceMethod(_person.class, @selector(coding));
Method curMethod = class_getInstanceMethod(_person.class, @selector(eating));
method_exchangeImplementations(oriMethod, curMethod);
交换⽅法的实现
4.5 拦截并替换⽅法
这个功能和上⾯的其实有些类似, 拦截并替换⽅法可以拦截并替换同⼀个类的, 也可以在两个类之间进⾏, 我这⾥⽤了两个不同的类, 下⾯是简单的代码实现.
_person = [NNPerson new];
_library = [NNLibrary new];
self.testLabelText = [_library libraryMethod];
Method oriMethod = class_getInstanceMethod(_person.class, @selector(changeMethod));
Method curMethod = class_getInstanceMethod(_library.class, @selector(libraryMethod));
method_exchangeImplementations(oriMethod, curMethod);
4.6 在⽅法上增加额外功能
这个使⽤场景还是挺多的, ⽐如我们需要记录 APP 中某⼀个按钮的点击次数, 这个时候我们便可以利⽤ runtime 来实现这个功能. 我这⾥写了个 UIButton 的⼦类, 然后在 + (void)load 中⽤ runtime 给它增加了⼀个功能, 核⼼代码及实现效果图如下:
+ (void)load {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
Method oriMethod = class_getInstanceMethod(self.class, @selector(sendAction:to:forEvent:));
Method cusMethod = class_getInstanceMethod(self.class, @selector(customSendAction:to:forEvent:));
// 判断⾃定义的⽅法是否实现, 避免崩溃
BOOL addSuccess = class_addMethod(self.class, @selector(sendAction:to:forEvent:), method_getImplementation(cusMethod), method_getTypeEncoding(cusMethod)); if (addSuccess) {
// 没有实现, 将源⽅法的实现替换到交换⽅法的实现
class_replaceMethod(self.class, @selector(customSendAction:to:forEvent:), method_getImplementation(oriMethod), method_getTypeEncoding(oriMethod));
} else {
// 已经实现, 直接交换⽅法
method_exchangeImplementations(oriMethod, cusMethod);
}
});
}
在⽅法上增加额外功能
4.7 归档解档
当我们使⽤ NSCoding 进⾏归档及解档时, 如果不⽤ runtime, 那么不管模型⾥⾯有多少属性, 我们都需要对其实现⼀遍 encodeObject 和decodeObjectForKey ⽅法, 如果模型⾥⾯有 10000 个属性, 那么我们就需要写 10000 句encodeObject 和 decodeObjectForKey ⽅法, 这个
时候⽤ runtime, 便可以充分体验其好处(以下只是核⼼代码, 具体代码请见 demo).
- (void)encodeWithCoder:(NSCoder *)aCoder {
unsigned int count = 0;
// 获取类中所有属性
Ivar *ivars = class_copyIvarList(self.class, &count);
// 遍历属性
for (int i = 0; i < count; i ++) {
// 取出 i 位置对应的属性
Ivar ivar = ivars[i];
// 查看属性
const char *name = ivar_getName(ivar);
NSString *key = [NSString stringWithUTF8String:name];
// 利⽤ KVC 进⾏取值,根据属性名称获取对应的值
id value = [self valueForKey:key];
[aCoder encodeObject:value forKey:key];
}
free(ivars);
}
- (instancetype)initWithCoder:(NSCoder *)aDecoder {
if (self = [super init]) {
unsigned int count = 0;
// 获取类中所有属性
Ivar *ivars = class_copyIvarList(self.class, &count);
// 遍历属性
for (int i = 0; i < count; i ++) {
// 取出 i 位置对应的属性
Ivar ivar = ivars[i];
// 查看属性
const char *name = ivar_getName(ivar);
NSString *key = [NSString stringWithUTF8String:name];
// 进⾏解档取值
id value = [aDecoder decodeObjectForKey:key];
// 利⽤ KVC 对属性赋值
[self setValue:value forKey:key];
}
}
return self;
}
4.8 字典转模型
字典转模型我们通常⽤的都是第三⽅, MJExtension, YYModel 等, 但也有必要了解⼀下其实现⽅式: 遍历模型中的所有属性,根据模型的属性名,去字典中查找对应的 key,取出对应的值,给模型的属性赋值。

/** 字典转模型 **/
+ (instancetype)modelWithDict:(NSDictionary *)dict {
id objc = [[self alloc] init];
unsigned int count = 0;
// 获取成员属性数组
Ivar *ivarList = class_copyIvarList(self, &count);
// 遍历所有的成员属性名
for (int i = 0; i < count; i ++) {
// 获取成员属性
Ivar ivar = ivarList[i];
// 获取成员属性名
NSString *ivarName = [NSString stringWithUTF8String:ivar_getName(ivar)];
NSString *key = [ivarName substringFromIndex:1];
// 从字典中取出对应 value 给模型属性赋值
id value = dict[key];
// 获取成员属性类型
NSString *ivarType = [NSString stringWithUTF8String:ivar_getTypeEncoding(ivar)];
// 判断 value 是不是字典
if ([value isKindOfClass:[NSDictionary class]]) {
ivarType = [ivarType stringByReplacingOccurrencesOfString:@"@" withString:@""];
ivarType = [ivarType stringByReplacingOccurrencesOfString:@"\"" withString:@""];
Class modalClass = NSClassFromString(ivarType);
// 字典转模型
if (modalClass) {
// 字典转模型
value = [modalClass modelWithDict:value];
}
}
if ([value isKindOfClass:[NSArray class]]) {
// 判断对应类有没有实现字典数组转模型数组的协议
if ([self respondsToSelector:@selector(arrayContainModelClass)]) {
// 转换成id类型,就能调⽤任何对象的⽅法
id idSelf = self;
// 获取数组中字典对应的模型
NSString *type = [idSelf arrayContainModelClass][key];
// ⽣成模型
Class classModel = NSClassFromString(type);
NSMutableArray *arrM = [NSMutableArray array];
// 遍历字典数组,⽣成模型数组
for (NSDictionary *dict in value) {
// 字典转模型
id model = [classModel modelWithDict:dict];
[arrM addObject:model];
}
// 把模型数组赋值给value
value = arrM;
}
}
// KVC 字典转模型
if (value) {
[objc setValue:value forKey:key];
}
}
return objc;
}
上⾯的所有代码都可以在这⾥下载:
总结
以上就是这篇⽂章的全部内容了,希望本⽂的内容对⼤家的学习或者⼯作具有⼀定的参考学习价值,如果有疑问⼤家可以留⾔交流,谢谢⼤家对的⽀持。

相关文档
最新文档