玩转iOS开发:装逼技术RunTime的总结篇

文章分享至个人我的技术博客: https://cainluo.github.io/15077238804616.htmlhtml


讲到这里, RunTime的知识点和应用基本上就已经讲完了, 剩下的就靠你们本身在项目中的应用以及累积了.git

这一篇文章主要是github

转载声明:如须要转载该文章, 请联系做者, 而且注明出处, 以及不能擅自修改本文.面试


RunTime基础

基础路径图:数组

1

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

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

咱们能够从这些东西里获取到指定类的全部信息, 不管是公开的, 仍是私有的, 所有均可以拿到, 而且操做.函数

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


RunTime进阶

进阶路径图:学习

2

在学习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应用

应用路径图:

3

在学习完RunTime以后, 咱们就能够应用到咱们的实际开发中.

Category

  • 关联对象
  • 控制对象

Class

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

Model

  • 自动归档和解档
  • 自动字典转模型
    • 字典转模型(模型属性数量大于字典key数量)
    • 字典转模型(模型中嵌套模型)
    • 字典转模型(数组中嵌套模型)

RunTime实例开发场景

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

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

Runtime Method Swizzling开发实例汇总


RunTime面试题及答案

问题: objc在向一个对象发送消息时, 发生了什么?
1.根据对象的isa指针找到类对象id, 在查询类对象里面的methodLists方法函数列表
2.若是没有在好到, 在沿着superClass, 寻找父类,再在父类methodLists方法列表里面查询
3.最终找到SEL, 根据idSEL确认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_setvarlayoutclass_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 : 属性列表.

总结

好了, 终于到尾声了, 但愿你们能够在个人文章里学到知识, 早日迎娶白富美, 走上人生巅峰, 最后附上全部的文章:

RunTime快速入门

RunTime应用


最后

码字很费脑, 看官赏点饭钱可好

微信

支付宝