基础路径图:

在学习 RunTime 的基础时, 我们要搞清楚一些重要的东西, 一些专业术语:

  • SEL
  • id
  • Class
  • Method
  • Ivar
  • IMP
  • Cache
  • Property

我们可以从这些东西里获取到指定类的所有信息, 无论是公开的, 还是私有的, 全部都可以拿到, 并且操作.

PS: 但操作私有方法的时候, 注意不要用来上架, 除非你有方法让苹果审核的时候通过.

RunTime进阶

进阶路径图:

作为一个开发者,有一个学习的氛围跟一个交流圈子特别重要,这是一个我的iOS交流群:642363427,不管你是大牛还是小白都欢迎入驻 ,分享BAT,阿里面试题、面试经验,讨论技术, 大家一起交流学习成长!

在学习 RunTime 进阶的时候, 我们就要了解更加的深入.

消息机制:

  • objc_msgSend
  • objc_msgSend_fpret
  • objc_msgSend_stret
  • objc_msgSendSuper
  • objc_msgSendSuper_stret

对象关联:

  • objc_setAssociatedObject()
  • objc_getAssociatedObject()
  • objc_removeAssociatedObjects()

对象关联的策略:

  • OBJC_ASSOCIATION_ASSIGN
  • OBJC_ASSOCIATION_RETAIN_NONATOMIC
  • OBJC_ASSOCIATION_COPY_NONATOMIC
  • OBJC_ASSOCIATION_RETAIN
  • OBJC_ASSOCIATION_COPY

动态方法解析:

  • resolveInstanceMethod:

    • YES , 通过 class_addMethod 消息得到处理, 结束
    • NO , 进入 forwardingTargetForSelector

      • 指定响应 selector , 消息得到处理, 结束
      • 不指定响应 selector

        • 进入 methodSignatureForSelector , 指定方法签名, 调用 forwardInvovation , 通过 anInvocation 做处理, 消息得到处理, 结束
        • 不指定方法签名, 该消息没有得到处理, 系统报错

RunTime应用

应用路径图:

在学习完 RunTime 之后, 我们就可以应用到我们的实际开发中.

Category

  • 关联对象
  • 控制对象

Class

  • 动态添加方法
  • 动态交换方法
  • 动态拦截并替换方法
  • 动态给方法添加额外功能

Model

  • 自动归档和解档
  • 自动字典转模型

    • 字典转模型(模型属性数量大于字典key数量)
    • 字典转模型(模型中嵌套模型)
    • 字典转模型(数组中嵌套模型)

RunTime实例开发场景

在实际开发中, 我们有一些实例场景会用到 RunTime :

  • 替换 ViewController 的声明周期
  • 解决集合类因索引的问题崩溃的问题
  • 防止按钮重复高强度点击
  • 全局更换控件初始效果
  • App热修复
  • App异常加载的展位图
  • 全局修改 UINavigationBar 的 backButtonItem

RunTime面试题及答案

问题: objc在向一个对象发送消息时, 发生了什么?
1.根据对象的 isa 指针找到类对象 id , 在查询类对象里面的 methodLists 方法函数列表
2.如果没有在好到, 在沿着 superClass , 寻找父类,再在父类 methodLists 方法列表里面查询
3.最终找到 SEL , 根据 id 和 SEL 确认 IMP (指针函数), 在发送消息.
问题: 什么时候会报 unrecognized selector 错误? iOS 有哪些机制来避免走到这一步?
1.当发送消息的时候, 我们会根据类里面的 methodLists 列表去查询我们要动用的 SEL , 当查询不到的时候, 我们会一直沿着父类查询
2.当最终查询不到的时候我们会报 unrecognized selector 错误, 当系统查询不到方法的时候, 会调用 +(BOOL)resolveInstanceMethod:(SEL)sel 动态解释的方法来给我一次机会来添加, 调用不到的方法.
3.或者我们可以再次使用 -(id)forwardingTargetForSelector:(SEL)aSelector 重定向的方法来告诉系统,该调用什么方法,一来保证不会崩溃.
问题: 能否向编译后得到的类中增加实例变量?能否向运行时创建的类中添加实例变量? 为什么?
1.不能向编译后得到的类增加实例变量.
2.能向运行时创建的类中添加实例变量.
解释:
1. 编译后的类已经注册在 runtime 中,类结构体中的 objc_ivar_list 实例变量的链表和 instance_size 实例变量的内存大小已经确定, runtime 会调用 class_setvarlayout 或 class_setWeaklvarLayout 来处理 strong`weak` 引用.所以不能向存在的类中添加实例变量.
2. 运行时创建的类是可以添加实例变量,调用 class_addIvar 函数. 但是的在调用 objc_allocateClassPair 之后, objc_registerClassPair 之前,原因同上.
问题: runtime如何实现weak变量的自动置nil?
1. runtime 对注册的类, 会进行布局,对于 weak 对象会放入一个 hash 表中。 用 weak 指向的对象内存地址作为 key ,当此对象的引用计数为 0 的时候会 dealloc .
2.假如 weak 指向的对象内存地址是 A ,那么就会以 A 为键, 在这个 weak 表中搜索,找到所有以 A为键的 weak 对象,从而设置为 nil .
问题: 给类添加一个属性后,在类结构体里哪些元素会发生变化
1. instance_size :实例的内存大小.
2. objc_ivar_list *ivars : 属性列表.