使用objc runtime实现闭环的懒加载

栏目: Objective-C · 发布时间: 5年前

内容简介:文章所指的懒加载形式通常如下一般使用宏定义可以轻松完成。但是没有一致性,移植差。 利用objc runtime的动态性实现懒加载可以实现

地址:AutoPropertyCocoa

文章所指的懒加载形式通常如下

- (id)lazyloadProperty{

    if(_lazyloadProperty == nil){

        _lazyloadProperty = [XClass ...];
    }
    return _lazyloadProperty;
}复制代码

一般使用宏定义可以轻松完成。但是没有一致性,移植差。 利用objc runtime的动态性实现懒加载可以实现 即可增加又可删除功能 ,可以针对单个实例进行修改,避免污染类型。该三方弥补了目前没有闭环实现懒加载三方的空缺。

主要实现:

  1. 实例或者类的懒加载
  2.  如果是实例对象则钩住并修改类型将其子类化
  3. 对该类型进行method swizzling 如果现在进行解绑,则判断是否是自己实现的方法.
  4. 如果是自己实现的方法->5,否则->6
  5. 调用method swizzling
  6. 还原 删除这个类型的这个方法

主要难点和 有价值 的内容 :

  1. 自己实现method swizzling
  2. 重新实现objc1时代的方法class_removeMethods
  3. 钩住运行时中的runtimelock,实现修改类型数据时的安全性

我们在实现method swizzling时的两个API

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时代,是可以通过下列方法删除的:

OBJC_EXPORT void
class_removeMethods(Class _Nullable, struct objc_method_list * _Nonnull)
    OBJC2_UNAVAILABLE;复制代码

Objc2时代之后runtime被重写后没有该方法了,并且新的runtime的类结构看起来就没打算让开发者删除方法,所以这里将过程记下。

首先看类读写器的结构class_rw_t

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_list_t.

method_list_t又继承于entsize_list_tt,他也是数组结构。

整个method_array_t结构是二维数组。每次删掉一个method_t需要用新method_list_t替换原对象。

使用objc runtime实现闭环的懒加载

然后是线程安全的问题,需要获取到苹果在操作类型的时候使用的读写锁(pthread_rw_lock_t runtimelock)。没有这把锁任何对runtime的修改都是不可靠的。

最终采取的方式是:劫持暴露了符号的系统函数然后阻塞线程

使用objc runtime实现闭环的懒加载

系统C函数使用的是脸书的鱼钩,这个钩子在macOS其实也是可以正常工作的。

剩下的就是寻找合适的函数了,这函数要满足两个条件:

  1. 该函数在符号表中存在
  2. 函数内部在lock runtimelock之后存在满足条件1的第二个函数

找了半天发现最合适的只有objc_allocateProtocol()了,objc_allocateProtocol内部会调用calloc(),所以第二个被劫持函数就是calloc。为了减小calloc的开销,需要稍微做一些工作。

  1. 对每次调用进行比较线程ID的操作显然比暴力阻塞线程好。
  2.  减小劫持后的calloc的调用栈。

虽然不是什么吸引人的UI框架还是希望大家点个赞吧。另外,有成都的公司招iOS吗:joy:?

meterwhite@outlook.com


以上所述就是小编给大家介绍的《使用objc runtime实现闭环的懒加载》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

程序的力量

程序的力量

甄贞 / 法律出版社 / 2002-3 / 21.00元

本书所谈及的话题概括了刑诉法学研究领域的方方面面,既有对每性、广泛性、前瞻性的宏观学科前沿问题的把握;又有实践性、直观性、详细性的个案分析和具体程序操作问题之探讨等。一起来看看 《程序的力量》 这本书的介绍吧!

XML、JSON 在线转换
XML、JSON 在线转换

在线XML、JSON转换工具

XML 在线格式化
XML 在线格式化

在线 XML 格式化压缩工具

HSV CMYK 转换工具
HSV CMYK 转换工具

HSV CMYK互换工具