维护三方 發表於 2020-8-15 21:59:00

iOS开发——RunTime总结篇

<h3 id="基础路径图">基础路径图:</h3>
<p><img src="//upload-images.jianshu.io/upload_images/16555213-7400cdc6730216ef.png!web?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240"></p>
<p>在学习&nbsp;<code>RunTime</code>&nbsp;的基础时, 我们要搞清楚一些重要的东西, 一些专业术语:</p>
<ul>
<li>SEL</li>
<li>id</li>
<li>Class</li>
<li>Method</li>
<li>Ivar</li>
<li>IMP</li>
<li>Cache</li>
<li>Property</li>
</ul>
<p>我们可以从这些东西里获取到指定类的所有信息, 无论是公开的, 还是私有的, 全部都可以拿到, 并且操作.</p>
<blockquote>
<p><strong>作为一个开发者,有一个学习的氛围跟一个交流圈子特别重要,这是一个我的iOS交流群:413038000,不管你是大牛还是小白都欢迎入驻 ,分享BAT,阿里面试题、面试经验,讨论技术, 大家一起交流学习成长!</strong></p>
</blockquote>
<h4 id="ps-但操作私有方法的时候-注意不要用来上架-除非你有方法让苹果审核的时候通过">PS: 但操作私有方法的时候, 注意不要用来上架, 除非你有方法让苹果审核的时候通过.</h4>
<h3 id="runtime进阶">RunTime进阶</h3>
<p>进阶路径图:</p>
<p><img src="//upload-images.jianshu.io/upload_images/16555213-f805b1f1ea3c99dc.png!web?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240"></p>
<p>在学习&nbsp;<code>RunTime</code>&nbsp;进阶的时候, 我们就要了解更加的深入.</p>
<h4 id="消息机制">消息机制:</h4>
<ul>
<li>objc_msgSend</li>
<li>objc_msgSend_fpret</li>
<li>objc_msgSend_stret</li>
<li>objc_msgSendSuper</li>
<li>objc_msgSendSuper_stret</li>
</ul>
<h4 id="对象关联">对象关联:</h4>
<ul>
<li>objc_setAssociatedObject()</li>
<li>objc_getAssociatedObject()</li>
<li>objc_removeAssociatedObjects()</li>
</ul>
<h4 id="对象关联的策略">对象关联的策略:</h4>
<ul>
<li>OBJC_ASSOCIATION_ASSIGN</li>
<li>OBJC_ASSOCIATION_RETAIN_NONATOMIC</li>
<li>OBJC_ASSOCIATION_COPY_NONATOMIC</li>
<li>OBJC_ASSOCIATION_RETAIN</li>
<li>OBJC_ASSOCIATION_COPY</li>
</ul>
<h4 id="动态方法解析">动态方法解析:</h4>
<ul>
<li><code>resolveInstanceMethod:</code>
<ul>
<li><code>YES</code>&nbsp;, 通过&nbsp;<code>class_addMethod</code>&nbsp;消息得到处理, 结束</li>
<li><code>NO</code>&nbsp;, 进入&nbsp;<code>forwardingTargetForSelector</code>
<ul>
<li>指定响应&nbsp;<code>selector</code>&nbsp;, 消息得到处理, 结束</li>
<li>不指定响应&nbsp;<code>selector</code>
<ul>
<li>进入&nbsp;<code>methodSignatureForSelector</code>&nbsp;, 指定方法签名, 调用&nbsp;<code>forwardInvovation</code>&nbsp;, 通过&nbsp;<code>anInvocation</code>&nbsp;做处理, 消息得到处理, 结束</li>
<li>不指定方法签名, 该消息没有得到处理, 系统报错</li>
</ul>
</li>
</ul>
</li>
</ul>
</li>
</ul>
<h3 id="runtime应用">RunTime应用</h3>
<p>应用路径图:</p>
<p><img src="//upload-images.jianshu.io/upload_images/16555213-eb1b032fbb339757.png!web?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240"></p>
<p>在学习完&nbsp;<code>RunTime</code>&nbsp;之后, 我们就可以应用到我们的实际开发中.</p>
<h4 id="category">Category</h4>
<ul>
<li>关联对象</li>
<li>控制对象</li>
</ul>
<h4 id="class">Class</h4>
<ul>
<li>动态添加方法</li>
<li>动态交换方法</li>
<li>动态拦截并替换方法</li>
<li>动态给方法添加额外功能</li>
</ul>
<h4 id="model">Model</h4>
<ul>
<li>自动归档和解档</li>
<li>自动字典转模型
<ul>
<li>字典转模型(模型属性数量大于字典key数量)</li>
<li>字典转模型(模型中嵌套模型)</li>
<li>字典转模型(数组中嵌套模型)</li>
</ul>
</li>
</ul>
<h3 id="runtime实例开发场景">RunTime实例开发场景</h3>
<p>在实际开发中, 我们有一些实例场景会用到&nbsp;<code>RunTime</code>&nbsp;:</p>
<ul>
<li>替换&nbsp;<code>ViewController</code>&nbsp;的声明周期</li>
<li>解决集合类因索引的问题崩溃的问题</li>
<li>防止按钮重复高强度点击</li>
<li>全局更换控件初始效果</li>
<li>App热修复</li>
<li>App异常加载的展位图</li>
<li>全局修改&nbsp;<code>UINavigationBar</code>&nbsp;的&nbsp;<code>backButtonItem</code></li>
</ul>
<p>Runtime Method Swizzling开发实例汇总</p>
<h3 id="runtime面试题及答案">RunTime面试题及答案</h3>
<table>
<thead>
<tr>
<th><strong>问题:</strong>&nbsp;objc在向一个对象发送消息时, 发生了什么?</th>
</tr>
</thead>
<tbody>
<tr>
<td>1.根据对象的&nbsp;<code>isa</code>&nbsp;指针找到类对象&nbsp;<code>id</code>&nbsp;, 在查询类对象里面的&nbsp;<code>methodLists</code>&nbsp;方法函数列表</td>
</tr>
<tr>
<td>2.如果没有在好到, 在沿着&nbsp;<code>superClass</code>&nbsp;, 寻找父类,再在父类&nbsp;<code>methodLists</code>&nbsp;方法列表里面查询</td>
</tr>
<tr>
<td>3.最终找到&nbsp;<code>SEL</code>&nbsp;, 根据&nbsp;<code>id</code>&nbsp;和&nbsp;<code>SEL</code>&nbsp;确认&nbsp;<code>IMP</code>&nbsp;(指针函数), 在发送消息.</td>
</tr>
</tbody>
</table>
<table>
<thead>
<tr>
<th><strong>问题:</strong>&nbsp;什么时候会报&nbsp;<code>unrecognized selector</code>&nbsp;错误?&nbsp;<code>iOS</code>&nbsp;有哪些机制来避免走到这一步?</th>
</tr>
</thead>
<tbody>
<tr>
<td>1.当发送消息的时候, 我们会根据类里面的&nbsp;<code>methodLists</code>&nbsp;列表去查询我们要动用的&nbsp;<code>SEL</code>&nbsp;, 当查询不到的时候, 我们会一直沿着父类查询</td>
</tr>
<tr>
<td>2.当最终查询不到的时候我们会报&nbsp;<code>unrecognized selector</code>&nbsp;错误, 当系统查询不到方法的时候, 会调用&nbsp;<code>+(BOOL)resolveInstanceMethod:(SEL)sel</code>&nbsp;动态解释的方法来给我一次机会来添加, 调用不到的方法.</td>
</tr>
<tr>
<td>3.或者我们可以再次使用&nbsp;<code>-(id)forwardingTargetForSelector:(SEL)aSelector</code>&nbsp;重定向的方法来告诉系统,该调用什么方法,一来保证不会崩溃.</td>
</tr>
</tbody>
</table>
<table>
<thead>
<tr>
<th><strong>问题:</strong>&nbsp;能否向编译后得到的类中增加实例变量?能否向运行时创建的类中添加实例变量? 为什么?</th>
</tr>
</thead>
<tbody>
<tr>
<td>1.不能向编译后得到的类增加实例变量.</td>
</tr>
<tr>
<td>2.能向运行时创建的类中添加实例变量.</td>
</tr>
<tr>
<td><strong>解释:</strong></td>
</tr>
<tr>
<td>1. 编译后的类已经注册在&nbsp;<code>runtime</code>&nbsp;中,类结构体中的&nbsp;<code>objc_ivar_list</code>&nbsp;实例变量的链表和&nbsp;<code>instance_size</code>&nbsp;实例变量的内存大小已经确定,&nbsp;<code>runtime</code>&nbsp;会调用&nbsp;<code>class_setvarlayout</code>&nbsp;或&nbsp;<code>class_setWeaklvarLayout</code>&nbsp;来处理&nbsp;<code>strong``weak</code>&nbsp;引用.所以不能向存在的类中添加实例变量.</td>
</tr>
<tr>
<td>2. 运行时创建的类是可以添加实例变量,调用&nbsp;<code>class_addIvar</code>&nbsp;函数. 但是的在调用&nbsp;<code>objc_allocateClassPair</code>&nbsp;之后,&nbsp;<code>objc_registerClassPair</code>&nbsp;之前,原因同上.</td>
</tr>
</tbody>
</table>
<table>
<thead>
<tr>
<th><strong>问题:</strong>&nbsp;runtime如何实现weak变量的自动置nil?</th>
</tr>
</thead>
<tbody>
<tr>
<td>1.&nbsp;<code>runtime</code>&nbsp;对注册的类, 会进行布局,对于&nbsp;<code>weak</code>&nbsp;对象会放入一个&nbsp;<code>hash</code>&nbsp;表中。 用&nbsp;<code>weak</code>&nbsp;指向的对象内存地址作为&nbsp;<code>key</code>&nbsp;,当此对象的引用计数为&nbsp;<code>0</code>&nbsp;的时候会&nbsp;<code>dealloc</code>&nbsp;.</td>
</tr>
<tr>
<td>2.假如&nbsp;<code>weak</code>&nbsp;指向的对象内存地址是&nbsp;<code>A</code>&nbsp;,那么就会以&nbsp;<code>A</code>&nbsp;为键, 在这个&nbsp;<code>weak</code>&nbsp;表中搜索,找到所有以&nbsp;<code>A</code>为键的&nbsp;<code>weak</code>&nbsp;对象,从而设置为&nbsp;<code>nil</code>&nbsp;.</td>
</tr>
</tbody>
</table>
<table>
<thead>
<tr>
<th><strong>问题:</strong>&nbsp;给类添加一个属性后,在类结构体里哪些元素会发生变化</th>
</tr>
</thead>
<tbody>
<tr>
<td>1.&nbsp;<code>instance_size</code>&nbsp;:实例的内存大小.</td>
</tr>
<tr>
<td>2.&nbsp;<code>objc_ivar_list *ivars</code>&nbsp;: 属性列表.</td>
</tr>
</tbody>
</table>
<h3 id="总结">总结</h3>
<p>好了, 终于到尾声了, 希望大家可以在我的文章里学到知识, 早日迎娶白富美, 走上人生巅峰,</p><br><br>
来源:https://www.cnblogs.com/iOSer1122/p/13510565.html
頁: [1]
查看完整版本: iOS开发——RunTime总结篇