OC中分类的实现原理

OC中为类添加一个分类(Category)可以实现为类添加对象方法、类方法、添加属性(添加的属性不生成成员变量)、遵守协议。那么:

  1. 分类(Category)是什么?
  2. 对象方法和类方法怎么实现在方法调用时覆盖类中定义的方法?
  3. 添加的属性怎么实现修改和访问?
  4. 分类遵守的协议时是在做什么?
一、分类(Category)的结构

我们写一个分类,通过命令行指令生成C++代码文件

cd Person+TestProperty.m所在文件夹
xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc Person+TestProperty.m

查看C++文件发现分类的数据类型是一个结构体_category_t, 并且一个分类就是一个结构体变量,即在编译OC生成C++代码时,一个分类就会变成一个结构体变量。

//  分类的数据类型:结构体_category_t
struct _category_t {
    const char *name;
    struct _class_t *cls;
    const struct _method_list_t *instance_methods;
    const struct _method_list_t *class_methods;
    const struct _protocol_list_t *protocols;
    const struct _prop_list_t *properties;
};
// 分类是一个结构体变量
static struct _category_t _OBJC_$_CATEGORY_Person_$_TestProperty __attribute__ ((used, section ("__DATA,__objc_const"))) = 
{
    "Person",
    0, // &OBJC_CLASS_$_Person,
    (const struct _method_list_t *)&_OBJC_$_CATEGORY_INSTANCE_METHODS_Person_$_TestProperty,
    (const struct _method_list_t *)&_OBJC_$_CATEGORY_CLASS_METHODS_Person_$_TestProperty,
    0,
    (const struct _prop_list_t *)&_OBJC_$_PROP_LIST_Person_$_TestProperty,
};
二、方法调用时覆盖类中定义的方法?

查看OC的源码

  1. 查找源码中category_t {, 可以验证分类是一个结构体,内部包含了对象方法列表instanceMethods和类方法列表classMethods
struct category_t {
    const char *name;
    classref_t cls;
    struct method_list_t *instanceMethods;
    struct method_list_t *classMethods;
    struct protocol_list_t *protocols;
    struct property_list_t *instanceProperties;
    // Fields below this point are not always present on disk.
    struct property_list_t *_classProperties;

    method_list_t *methodsForMeta(bool isMeta) {
        if (isMeta) return classMethods;
        else return instanceMethods;
    }

    property_list_t *propertiesForMeta(bool isMeta, struct header_info *hi);
};
  1. 因为方法覆盖肯定会对这个结构体中instanceMethods操作,搜索可找到如下代码:
    分类方法添加到类.png
    while (i--) {
        auto& entry = cats->list[i];

        method_list_t *mlist = entry.cat->methodsForMeta(isMeta);
        if (mlist) {
            mlists[mcount++] = mlist;
            fromBundle |= entry.hi->isBundle();
        }

        property_list_t *proplist = 
            entry.cat->propertiesForMeta(isMeta, entry.hi);
        if (proplist) {
            proplists[propcount++] = proplist;
        }

        protocol_list_t *protolist = entry.cat->protocols;
        if (protolist) {
            protolists[protocount++] = protolist;
        }
    }

    auto rw = cls->data();

    prepareMethodLists(cls, mlists, mcount, NO, fromBundle);
    rw->methods.attachLists(mlists, mcount);
    free(mlists);

// cls->data()
    class_rw_t* data() {
        return (class_rw_t *)(bits & FAST_DATA_MASK);
    }

追踪代码里面可以发现:

  • 分类的对象方法列表会添加到类对象的方法列表(二维数组)中;
  • 添加的方式是整个分类的方法列表添加到类对象的方法列表中作为一个元素;
  • 越在后面编译的分类就越先添加到mlists中;(因为while (i--)从后往前添加)。
  • 遍历所有分类,方法都添加到mlists中之后,把mlists经过attachLists函数处理放进类的class_rw_t成员变量methods中,类本身的方法列表会放到最后面。所以在方法调用时,从类的class_rw_t的方法列表methods中查找方法是从第一个元素开始查找,第一个元素也就是从最先添加(最后编译的那个分类)到里面的那个分类的方法列表,找到后会得到执行而后面的列表中有这个方法也会得不到执行,实现了方法的覆盖。
  1. 分类的方法列表什么时候添加的?这个就是顺着上述代码的调用查看:

_objc_init -> map_2_images -> map_images_nolock -> _read_images
-> 上图中显示的位置,进行分类信息合并到类信息中的操作。

_objc_init函数是源头,查看这个函数的解释:
可阅读文章 iOS底层探索之_objc_init

_objc_init是在APP冷启动时调用的,dyld加载所有文件和动态库中符号和调用类和分类load方法,所以我们知道了,在启动时runtime环境初始化的时候,分类的方法列表就会合并到类中方法列表里面。
程序冷启动的过程:
iOS 冷启动
iOS 程序 main 函数之前发生了什么

三、分类中的属性怎么实现存取的?

从源码中,可以看到分类的属性也是像方法列表一样添加到类的属性列表中(遵守的协议也是一样),分类中@Property声明的属性是不会生成成员变量的。
通过属性关联,见博客:OC属性关联的实现原理

四、分类中的协议做什么的,为什么类中要用数组保存遵守的协议?

遵守一个协议即拥有协议中方法的声明,保存类遵守的协议,比如有如下判断则需要到协议列表中查找。

BOOL isConforms = [self conformsToProtocol:NSProtocolFromString(@"UITableViewDelegate")];
五、类扩展与分类的区别?
  • 类扩展是在编译时就会将方法和属性合并到类中的,类扩展中声明属性和方法跟类是一样的效果。
  • 分类中的方法是在冷启动时添加到类的方法列表中的,分类中声明的属性不会生成成员变量,分类中的属性需要通过属性关联来实现存取属性值。
附:
  • 类class的构成如下,class_rw_t存在于bits当中。
struct objc_class : objc_object {
    // Class ISA;
    Class superclass;
    cache_t cache;             // 方法缓存
    class_data_bits_t bits;    // bits的内容就是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;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351