前言

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];
 person.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 拦截并替换方法

这个功能和上面的其实有些类似, 拦截并替换方法可以拦截并替换同一个类的, 也可以在两个类之间进行, 我这里用了两个不同的类, 下面是简单的代码实现.

 _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;
}

上面的所有代码都可以在这里下载: runtime 练习: NNRuntimeTest

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对Devmax的支持。

iOS开发中runtime常用的几种方法示例总结的更多相关文章

  1. 基于EJB技术的商务预订系统的开发

    用EJB结构开发的应用程序是可伸缩的、事务型的、多用户安全的。总的来说,EJB是一个组件事务监控的标准服务器端的组件模型。基于EJB技术的系统结构模型EJB结构是一个服务端组件结构,是一个层次性结构,其结构模型如图1所示。图2:商务预订系统的构架EntityBean是为了现实世界的对象建造的模型,这些对象通常是数据库的一些持久记录。

  2. HTML5在微信内置浏览器下右上角菜单的调整字体导致页面显示错乱的问题

    HTML5在微信内置浏览器下,在右上角菜单的调整字体导致页面显示错乱的问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

  3. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  4. ios – containerURLForSecurityApplicationGroupIdentifier:在iPhone和Watch模拟器上给出不同的结果

    我使用默认的XCode模板创建了一个WatchKit应用程序.我向iOSTarget,WatchkitAppTarget和WatchkitAppExtensionTarget添加了应用程序组权利.(这是应用程序组名称:group.com.lombax.fiveminutes)然后,我尝试使用iOSApp和WatchKitExtension访问共享文件夹URL:延期:iOS应用:但是,测试NSURL

  5. ios – Testflight无法安装应用程序

    我有几个测试人员注册了testflight并连接了他们的设备……他们有不同的ios型号……但是所有这些都有同样的问题.当他们从“safari”或“testflight”应用程序本身单击应用程序的安装按钮时……达到约90%并出现错误消息…

  6. ibm-mobilefirst – 在iOS 7.1上获取“无法安装应用程序,因为证书无效”错误

    当我的客户端将他们的设备更新到iOS7.1,然后尝试从AppCenter更新我们的应用程序时,我收到了上述错误.经过一番搜索,我找到了一个类似问题的帖子here.但是后来因为我在客户端使用AppCenter更新应用程序的环境中,我无法使用USB插件并为他们安装应用程序.在发布支持之前,是否有通过AppCenter进行下载的解决方法?

  7. ios – 视图的简单拖放?

    我正在学习iOS,但我找不到如何向UIView添加拖放行为.我试过了:它说“UIView没有可见的接口声明选择器addTarget”此外,我尝试添加平移手势识别器,但不确定这是否是我需要的它被称为,但不知道如何获得事件的坐标.在iOS中注册移动事件回调/拖放操作的标准简单方法是什么?

  8. ios – 什么控制iTunes中iPhone应用程序支持的语言列表?

    什么控制iPhone应用程序的iTunes页面中支持的语言?

  9. ios – 获得APNs响应BadDeviceToken或Unregistered的可能原因是什么?

    我知道设备令牌在某些时候是有效的.用户如何使其设备令牌变坏?从关于“未注册”的文档:Thedevicetokenisinactiveforthespecifiedtopic.这是否意味着应用程序已被删除?.您应该看到四种分发方法:如果您选择AppStore或Enterprise,您将在后面的对话框中看到Xcode将APNS权利更改为生产:如果选择AdHoc或Development,则aps-environment下的文本将是开发,然后应与后端的配置匹配.

  10. ios – 当我关闭应用程序时,我从调试器获得消息:由于信号15而终止

    我怎么能解决这个问题,我不知道这个链接MypreviousproblemaboutCoredata对我的问题有影响吗?当我cmd应用程序的Q时,将出现此消息.Messagefromdebugger:Terminatedduetosignal15如果谁知道我以前的问题的解决方案,请告诉我.解决方法>来自调试器的消息:每当用户通过CMD-Q(退出)或STOP手动终止应用程序(无论是在iOS模拟器中还是

随机推荐

  1. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  2. iOS – genstrings:无法连接到输出目录en.lproj

    使用我桌面上的项目文件夹,我启动终端输入:cd然后将我的项目文件夹拖到终端,它给了我路径.然后我将这行代码粘贴到终端中找.-name*.m|xargsgenstrings-oen.lproj我在终端中收到此错误消息:genstrings:无法连接到输出目录en.lproj它多次打印这行,然后说我的项目是一个目录的路径?没有.strings文件.对我做错了什么的想法?

  3. iOS 7 UIButtonBarItem图像没有色调

    如何确保按钮图标采用全局色调?解决方法只是想将其转换为根注释,以便为“回答”复选标记提供更好的上下文,并提供更好的格式.我能想出这个!

  4. ios – 在自定义相机层的AVFoundation中自动对焦和自动曝光

    为AVFoundation定制图层相机创建精确的自动对焦和曝光的最佳方法是什么?

  5. ios – Xcode找不到Alamofire,错误:没有这样的模块’Alamofire’

    我正在尝试按照github(https://github.com/Alamofire/Alamofire#cocoapods)指令将Alamofire包含在我的Swift项目中.我创建了一个新项目,导航到项目目录并运行此命令sudogeminstallcocoapods.然后我面临以下错误:搜索后我设法通过运行此命令安装cocoapodssudogeminstall-n/usr/local/bin

  6. ios – 在没有iPhone6s或更新的情况下测试ARKit

    我在决定下载Xcode9之前.我想玩新的框架–ARKit.我知道要用ARKit运行app我需要一个带有A9芯片或更新版本的设备.不幸的是我有一个较旧的.我的问题是已经下载了新Xcode的人.在我的情况下有可能运行ARKit应用程序吗?那个或其他任何模拟器?任何想法或我将不得不购买新设备?解决方法任何iOS11设备都可以使用ARKit,但是具有高质量AR体验的全球跟踪功能需要使用A9或更高版本处理器的设备.使用iOS11测试版更新您的设备是必要的.

  7. 将iOS应用移植到Android

    我们制作了一个具有2000个目标c类的退出大型iOS应用程序.我想知道有一个最佳实践指南将其移植到Android?此外,由于我们的应用程序大量使用UINavigation和UIView控制器,我想知道在Android上有类似的模型和实现.谢谢到目前为止,guenter解决方法老实说,我认为你正在计划的只是制作难以维护的糟糕代码.我意识到这听起来像很多工作,但从长远来看它会更容易,我只是将应用程序的概念“移植”到android并从头开始编写.

  8. ios – 在Swift中覆盖Objective C类方法

    我是Swift的初学者,我正在尝试在Swift项目中使用JSONModel.我想从JSONModel覆盖方法keyMapper,但我没有找到如何覆盖模型类中的Objective-C类方法.该方法的签名是:我怎样才能做到这一点?解决方法您可以像覆盖实例方法一样执行此操作,但使用class关键字除外:

  9. ios – 在WKWebView中获取链接URL

    我想在WKWebView中获取tapped链接的url.链接采用自定义格式,可触发应用中的某些操作.例如HTTP://我的网站/帮助#深层链接对讲.我这样使用KVO:这在第一次点击链接时效果很好.但是,如果我连续两次点击相同的链接,它将不报告链接点击.是否有解决方法来解决这个问题,以便我可以检测每个点击并获取链接?任何关于这个的指针都会很棒!解决方法像这样更改addobserver在observeValue函数中,您可以获得两个值

  10. ios – 在Swift的UIView中找到UILabel

    我正在尝试在我的UIViewControllers的超级视图中找到我的UILabels.这是我的代码:这是在Objective-C中推荐的方式,但是在Swift中我只得到UIViews和CALayer.我肯定在提供给这个方法的视图中有UILabel.我错过了什么?我的UIViewController中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

返回
顶部