iOS底层原理 - Category实现原理(一)

通过探索Category底层原理回答以下问题

1) Category是否可以添加方法、属性、成员变量?Category是否可以遵守Protocol?

2) Category的本质是什么,在底层是怎么存储的?

3) Category的实现原理是什么,Catagory中的方法是如何调用到的?

4) Category中是否有Load方法,load方法是什么时候调用的?

5) load、initialize的区别

Category可以直接添加 属性、成员变量吗?

创建一个ZHPerson类



添加分类


发现分类中可以添加属性,方法,协议,但是不能添加成员变量。

分析为什么不能添加成员变量?

Category的底层数据结构

    首先创建两个分类 协助测试






将分类文件编译为.cpp文件,切换到文件所在文件夹下执行:

xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc ZHPerson+Sport.m

xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc ZHPerson+Eat.m


检索_I_ZHPerson_Sport_sport 和 _C_ZHPerson_Sport_sport


分析可知:上述代码创建了_method_list_t类型的结构体变量

_OBJC_$_CATEGORY_INSTANCE_METHODS_ZHPerson_$_Sport 

 _OBJC_$_CATEGORY_CLASS_METHODS_ZHPerson_$_Sport

分别用于存储实例方法列表和类方法列表



这里简单说下_objc_method中的method_type,详细介绍后续在探究runtime消息机制时再继续扒。method_type其实可以看做用字符缩写来表达的函数类型字符串,比如 v@:i 就是返回类型为void,第一个参数为id类型,第二个参数为指针类型,第三个参数为int类型的函数,如- (void)addStepCount:(int)count(我们知道iOS中方法调用会默认传入隐式参数 方法调用者:self 和 方法名:_cmd。这也是为什么我们可以在方法内部访问self、cmd的原因)

继续检索_OBJC_$_CATEGORY_ZHPerson_$_Sport


回答第二个问题:分类底层如何存储的

分析可知:上述代码创建了一个_category_t类型的结构体变量 _OBJC_$_CATEGORY_ZHPerson_$_Sport;并且传入了类方法列表、实例方法列表、协议类别、属性列表,具备了Category的所有信息,没错Category在底层就是_category_t类型,下边我们检索结构体类型_category_t,看下_category_t的定义;


到此我们已经清楚的看到了Category在底层的存储结构,并且可以看到底层并没有存储成员变量,这也就是为什么直接添加成员变量会报错的原因。

Category中的属性

并且我们知道在类中添加一个属性,系统为我们做了三件事

@property (nonatomic, copy) NSString *name;

    1) 创建了一个成员变量_name  

    2) 生成了setter、getter方法的声明

    3) 生成了setter、getter方法的实现

对比 ZHPerson 编译后的.cpp中的属性和分类中的区别

ZHPerson.cpp:


ZHPerson (Sport).cpp 文件


发现原类中生成了属性的settter 和getter 方法,但是分类中没有属性的settter 和getter 实现。

在Category添加属性系统仅仅只生成了setter、getter方法的声明

如何使Category中的属性与类中的属性具备同样的效果(关联对象)

    

可以通过runtime中的关联对象方法来实现


关联策略,和@property后的关键字对应.


系统是如何管理关联对象的

    去官网下载runtime源码,搜索objc_setAssociatedObject方法,这里不做过多分析,简单说下结论,后续会单开一篇扒关联对象的实现原理。

runtime用四个类来管理关联对象,AssociationsManager、AssociationsHashMap、AssociationsMap、ObjectAssociation

1)关联对象并不是存储在被关联对象本身内存中

2) 关联对象存储在全局的统一的一个AssociationsManager中

3)设置关联对象为nil,就相当于是移除关联对象

Category中的方法调用顺序 - 表象

再创建一个ZHPerson的子类ZHStudent ,再编写一个ZHStudent的分类,分类里书写life方法




结论:1、分类中方法会覆盖原类中的方法

2.、Compile Sources中编译顺序在后面的文件优先级会更高。

ZHPerson有两个分类Sport 和Eat,并且两个分类都实现了life方法,

但是全部调用的分类Sport的方法,和Build Phases 里的Compile Sources 里的文件编译顺序有关


1、分类中的方法优先级高于原类中的方法

2、后编译的分类优先级高于先编译的分类

3、我们常说的分类方法覆盖原类方法并不是真正的覆盖,只是objc_msgSend在分类中找到方法实现后不再继续查找。

Category方法调用顺序 - 本质

    1)OC中的方法调用简单的说就是通过实例对象(或类对象)的isa指针和类对象(或元类对象)的superClass指针去类(或元类)对象中查找方法。

    2)Category中的方法、属性等编译后是存储在category_t结构体中的,也就是说编译后分类中的方法并没有合并到类(或元类)中,我们是无法在类对象(或元类对象)中找到Category中的方法的。

    3)但是最终调用的时候我们却可以通过isa和superClass指针找到这些方法。所以我们有理由猜测runtime帮我们做了方法合并

    4)_objc_init就是runtime的初始化函数,是在app启动过程的"初始化除可执行文件外的所有Mach-O文件初始化调用的;按文件编译倒序将各分类中的方法、协议、属性列表分别整合成一个二维数组后,添加到原类中的方法列表,属性列表,协议列表。(分类中的方法属性等系统是什么时候如何添加到原类中的?)



Category中的+load方法

在创建的两个类文件以及3个分类文件添加+load方法,也添加上initialize()方法,方便后面测试。



不导入上面相关任何文件,不创建对象,直接运行

可以发现未做任何调用和对象创建的情况下,也会执行+ (void)load方法

尝试在xcode->targets->build phases->compile sources中调整文件编译顺序,发现

1、父类中的load优先于子类中调用,且不受编译顺序影响。

2、原类中的load方法优先于分类调用,且不受编译顺序影响。

3、两个分类中的load方法执行顺序根据编译顺序,且与其继承的父类无关系。


Category中的+ (void)initialize方法

再创建一个新类ZHDog,作对比


像测试load方法一样,不导入创建的任何文件不创建对象,直接运行但是并没有调用initialize方法。

创建对象

情况1

情况2

情况3


以上3种情况的打印结果都是下面结果:


3种情况编译顺序都是



说明同一个类中的initialize方法在多次创建对象时仅调用一次


现在只调整文件编译顺序:



打印结果没变

initialize调用总结:

1)父类调用优先级高于子类,不受编译顺序影响;

2) 分类会覆盖原类中的方法

注意,如果子类及子类分类没有实现initialize方法,根据runtime消息发送机制,父类中的initialize会调用两次

现在注释掉ZHStudent及其分类的initialize方法,其他全不变





load是只要类所在的文件被引用就会被调用,而initialize在类或其子类的第一个方法调用之前被调用(runtime 中load方法不能认为第一个方法)。load在main函数之前调用,initialize在main函数之后调用。这两个方法会被自动调用。

· load和initialize方法都不用显示的调用父类的方法而是自动调用,即使子类没有initialize方法也会调用父类的方法如果子类显示调用[super initialize],则父类多次调用,load方法则不会调用父类。

·load方法通常用来进行Method Swizzle,initialize方法一般用于初始化全局变量或静态变量。

·load和initialize方法内部使用了锁,因此它们是线程安全的。实现时要尽可能保持简单,避免阻塞线程,不要再使用锁。

·每个类只调用initialize一次。如果希望为类和类的类别执行独立初始化,则应该实现load方法

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容