iOS开发资料收集区iOS高质量博客iOS开发笔录

玩转iOS开发:装逼技术RunTime

2018-03-29  本文已影响1406人  梦想编程家小枫

RunTime基础

基础路径图:

这是我的iOS技术交流群659170228,欢迎加入一起交流共同进步

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

SEL

id

Class

Method

Ivar

IMP

Cache

Property

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

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

RunTime进阶

进阶路径图:

在学习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 Method Swizzling开发实例汇总

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: 属性列表.

总结

好了, 终于到尾声了, 希望大家可以在我的文章里学到知识, 早日迎娶白富美, 走上人生巅峰.

有一句话叫做三人行必有我师,其实做为一个开发者,有一个学习的氛围跟一个交流圈子特别重要。这是我的iOS技术交流群 659170228 ,欢迎加入一起交流共同进步,需要runtime的资料也可以加小编的qq3156033240,小编免费分享给大家。

上一篇下一篇

猜你喜欢

热点阅读