地址:autopropertycocoa
懒加载形式如下
1
2
3
4
5
6
7
8
|
- (id)lazyloadproperty{
if (_lazyloadproperty == nil){
_lazyloadproperty = [xclass ...];
}
return _lazyloadproperty;
}
|
一般使用宏定义可以轻松完成。但是没有一致性,移植差。
利用objc runtime的动态性实现懒加载可以实现即可增加又可删除功能,也可以避免污染类型。该三方弥补了目前没有闭环实现懒加载三方的空缺。
主要流程:
- 实例或者类的懒加载
- 如果是实例对象则钩住并修改类型将其子类化
- 对该类型进行method swizzling
- 如果现在进行解绑,则判断是否是自己实现的方法.如果是自己实现的方法->5,否则->6
- 调用method swizzling还原
- 删除这个类型的这个方法
难点:
自己实现method swizzling
- 重新实现objc1时代的方法class_removemethods
- 钩住运行时中的runtimelock,实现修改类型数据时的安全性
我们再实现method swizzling时的两个api
1
2
3
4
5
6
7
8
|
objc_export imp _nullable
class_replacemethod( class _nullable cls, sel _nonnull name, imp _nonnull imp,
const char * _nullable types)
objc_available(10.5, 2.0, 9.0, 1.0, 2.0);
objc_export void
method_exchangeimplementations(method _nonnull m1, method _nonnull m2)
objc_available(10.5, 2.0, 9.0, 1.0, 2.0);
|
不管使用哪种,如果这个类型没有实现该方法而是父类实现的话,就需要动态增加一个方法。动态增加的方法在objc1时代,是可以通过下列方法删除的:
1
2
3
|
objc_export void
class_removemethods( class _nullable, struct objc_method_list * _nonnull)
objc2_unavailable;
|
objc2时代之后runtime被重写后没有该方法了,并且新的runtime的类结构看起来就没打算让开发者删除方法,所以这里将过程记下。
首先看类读写器的结构class_rw_t
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
struct class_rw_t {
// be warned that symbolication knows the layout of this structure.
uint32_t flags;
uint32_t version;
const class_ro_t *ro;
method_array_t methods; //删除这里的一个方法
property_array_t properties;
protocol_array_t protocols;
class firstsubclass;
class nextsiblingclass;
char *demangledname;
#if support_indexed_isa
uint32_t index;
#endif
};
|
method_array_t继承于list_array_tt<method_t, method_list_t>
,它是数组结构。存储的内容是method_list_t.
method_list_t又继承于entsize_list_tt<method_t, method_list_t, 0x3>
,他也是数组结构。
整个method_array_t结构是二维数组。每次删掉一个method_t需要用新method_list_t替换原对象。
然后是线程安全的问题,需要获取到苹果在操作类型的时候使用的读写锁(pthread_rw_lock_t runtimelock)。没有这把锁任何对runtime的修改都是不可靠的。
最终采取的方式是:劫持暴露了符号的系统函数然后阻塞线程
劫持系统c函数使用的是脸书的鱼钩,这个钩子在macos其实也是可以正常工作的。
剩下的就是寻找合适的函数了,这函数要满足两个条件:
- 该函数在符号表中存在
- 函数内部在lock runtimelock之后存在满足条件1的第二个函数
找了半天发现最合适的只有objc_allocateprotocol()了,objc_allocateprotocol内部会调用calloc(),所以第二个被劫持函数就是calloc。为了减小calloc的开销,需要稍微做一些工作。
- 对每次调用进行比较线程id的操作显然比暴力阻塞线程好。
- 减小劫持后的calloc的调用栈。
总结
以上所述是小编给大家介绍的使用objc runtime实现ios闭环的懒加载功能,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,小编会及时回复大家的!
相关文章
- ASP.NET自助建站系统中如何实现多语言支持? 2025-06-10
- 64M VPS建站:如何选择最适合的网站建设平台? 2025-06-10
- ASP.NET本地开发时常见的配置错误及解决方法? 2025-06-10
- ASP.NET自助建站系统的数据库备份与恢复操作指南 2025-06-10
- 个人网站服务器域名解析设置指南:从购买到绑定全流程 2025-06-10
- 2025-07-10 怎样使用阿里云的安全工具进行服务器漏洞扫描和修复?
- 2025-07-10 怎样使用命令行工具优化Linux云服务器的Ping性能?
- 2025-07-10 怎样使用Xshell连接华为云服务器,实现高效远程管理?
- 2025-07-10 怎样利用云服务器D盘搭建稳定、高效的网站托管环境?
- 2025-07-10 怎样使用阿里云的安全组功能来增强服务器防火墙的安全性?
快网idc优惠网
QQ交流群
-
2025-05-25 95
-
2025-06-04 82
-
2025-05-27 36
-
2025-06-04 69
-
2025-05-25 75