博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
runtime 的常用姿势
阅读量:7119 次
发布时间:2019-06-28

本文共 6628 字,大约阅读时间需要 22 分钟。

一、介绍

runtime简称运行时,就是系统在运行时的一些机制,其中最主要的是消息机制。对于C语言,函数调用在编译的时候会决定调用哪个函数,编译完成之后直接顺序执行,无任何二义性。OC的函数调用为消息发送,属于动态调用过程,在编译的时候并不能决定真正调用拿个函数,只有在真正运行的时候才会根据函数名称去寻找对应的函数来调用。

二、常用函数

// 遍历某个类所有的成员变量class_copyIvarList// 遍历某个类所有的方法class_copyMethodList// 获取指定名称的成员变量class_getInstanceVariable// 获取成员变量名ivar_getName// 获取成员变量类型编码ivar_getTypeEncoding// 获取某个对象成员变量的值object_getIvar// 设置某个对象成员变量的值object_setIvar// 给对象发送消息objc_msgSend复制代码

三、相关应用方向

  • 更改对象属性
  • 动态添加属性
  • 动态添加方法
  • 交换方法的实现
  • 归档解档
  • 字典转模型
  • 访问私有变量

四、具体使用

在使用 runtime 的时候,需要先引入头文件

#import 
复制代码

4.1 更改属性

使用到的方法包括

// 遍历某个类所有的成员变量class_copyIvarList// 设置某个对象成员变量的值object_setIvar复制代码

用 runtime 修改一个对象的属性值

unsigned int count = 0;Person *person = [Person new];person.name = @"李白";    // 动态获取类中的所有属性(包括私有)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 可以实现给分类添加属性。下面就给 NSObject 分类添加属性:

// NSObject+Category.h 文件@interface NSObject(Category)@property (nonatomic, copy) NSString *name; //给分类添加属性@end// NSObject+Category.m 文件#import "NSObject+Category.h"#import 
static NSString *nameKey = @"nameKey"; // 定义一个key@implementation NSObject (Category)// runtime实现set方法- (void)setName:(NSString *)name { objc_setAssociatedObject(self, nameKey, name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);}// runtime实现get方法- (NSString *)name { return objc_getAssociatedObject(self, nameKey);}复制代码

这样我们在引用 NSObject+Category 后,创建的 NSObject 对象就会多处一个 name 属性了。

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 归档解档

当我们使用 NSCoding 进行归档及解档时, 如果不用 runtime, 那么不管模型里面有多少属性, 我们都需要对其实现一遍 encodeObject 和 decodeObjectForKey 方法, 如果模型里面有 10000 个属性, 那么我们就需要写 10000 句encodeObject 和 decodeObjectForKey 方法, 这个时候用 runtime, 便可以充分体验其好处,使用如下:

- (void)encodeWithCoder:(NSCoder *)coder {    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];        [coder encodeObject:value forKey:key];    }    free(ivars);}- (instancetype)initWithCoder:(NSCoder *)decoder {    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 = [decoder decodeObjectForKey:key];            // 利用 KVC 对属性赋值            [self setValue:value forKey:key];        }    }    return self;}复制代码

4.6 字典转模型

字典转模型我们通常用的都是第三方, 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;}复制代码

4.7 访问私有变量

我们知道,OC中没有真正意义上的私有变量和方法,要让成员变量私有,要放在m文件中声明,不对外暴露。如果我们知道这个成员变量的名称,可以通过runtime获取成员变量,再通过getIvar来获取它的值。方法:

Ivar ivar = class_getInstanceVariable([Model class], "_str1");NSString * str1 = object_getIvar(model, ivar);复制代码

参考了

转载于:https://juejin.im/post/5b6a8f50f265da0f894b9453

你可能感兴趣的文章
SoapUI Pro Project Solution Collection-DataSource(jdbc,excel)
查看>>
全国主要城市不同日照标准的间距系数
查看>>
python网络爬虫(一):网络爬虫科普与URL含义
查看>>
UVA 11732 - strcmp() Anyone?(Trie)
查看>>
Vue v-bind的使用
查看>>
36.5. height / width
查看>>
动手实践虚拟网络 - 每天5分钟玩转 OpenStack(10)
查看>>
【Python】supervisor 工具介绍
查看>>
浅谈嵌入式软件的未来发展
查看>>
Spark源码分析之二:Job的调度模型与运行反馈
查看>>
C#——await与async实现多线程异步编程
查看>>
如何找到一个好的Joomla主机提供商
查看>>
Dockerfile最佳实践(二)
查看>>
T-SQL Enhancement in SQL Server 2005[下篇]
查看>>
杀毒软件可能令企业用户陷入更大危机
查看>>
澳政府投资光伏发电 内外资项目角逐高额补助
查看>>
《从问题到程序:用Python学编程和计算》——2.4 字符串
查看>>
《AngularJS实战》——3.2 过滤器的应用
查看>>
《贝叶斯思维:统计建模的Python学习法》——2.5 封装框架
查看>>
《Cisco安全防火墙服务模块(FWSM)解决方案》——2.7 软件架构
查看>>