iOS Category、 Load 、Initialize笔记

一、Category分类

一个类永远只有一个类对象。
运行起来后,最后对象方法统一都会放在类对象中。如果存在类方法,那么统一都会放在元类方法中。
分类的合并是,运行时通过runtime动态的将分类的方法合并到类方法和元类方法中。

类和分类同时实现一个方法,会优先实现分类的,并不是覆盖,多个分类同时实现一个方法,则会首先实现编译在最后面的文件的方法。

分类不可以直接添加成员变量,可以间接的方式。

#import "Person.h"
 
@interface Person (Test)
 
@property (nonatomic, assign) int weight;
@property (nonatomic, assign) int name;
 
@end
 
@implementation Person (Test)
 
// 方案一
const void *weightKey = &weightKey;
 
- (void)setWeight:(int)weight
{
    objc_setAssociatedObject(self, weightKey, @(weight), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
 
- (int)weight
{
    return [objc_getAssociatedObject(self, weightKey) intValue];
}
 
 
 // 方案二
 static const char nameKey;
 
 
 - (void)setName:(int)name
 {
 objc_setAssociatedObject(self, &nameKey, @(name), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
 }
 
 - (int)name
 {
 return [objc_getAssociatedObject(self, &nameKey) intValue];
 }
 
 
 // 方案三
 #define nameKey @"name"
 
 - (void)setName:(int)name
 {
 // 这其实传进去的是字符串地址 : NSString *str = @"name";  @"name"放在数据常量区
 objc_setAssociatedObject(self, nameKey, @(name), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
 }
 
 - (int)name
 {
 return [objc_getAssociatedObject(self, nameKey) intValue];
 }
 
 
 
// 方案四
- (void)setName:(int)name
{
    //_cmd == @selector(name)
    //@selector(name) 相当于返回某个结构体的指针
    NSLog(@"%p %p %p", @selector(name), @selector(name), @selector(name));
    // 这其实传进去的是字符串地址 : NSString *str = @"name";  @"name"放在数据常量区
    objc_setAssociatedObject(self, @selector(name), @(name), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    // 上面第一个self只要是对象就行 因为接收的是id类型 并且这个方法不会影响person这类原来的内存结构,name也不会存进到perosn的ivar列表里
}
 
- (int)name
{
    // 这个也可以这么写    return [objc_getAssociatedObject(self, _cmd) intValue]; 因为前面已经设置了set方法,所以现在才可以用_cmd
    return [objc_getAssociatedObject(self, @selector(name)) intValue];
}
 
 
@end

二、load

不管用不用得到建立在项目中的类,都会被加载入内存。
load方法调用的时机:runtime在加载这个类、分类,就用调用对应的+load方法。
分类已经实现了load,类的load依然会被调用。
调用顺序为先调用类中的load,再带哦用分类中的load,与编译顺序无关。
load方法内部是重调用了load方法,并不是去调用的原来的类中直接调用的,所以会load方法不覆盖。

总结:
+load方法会在runtime加载类、分类时调用。
每个类、分类的+load,在程序运行过程中只调用一次。
调用顺序:
1:先调用类的+load;
a:按照编译先后顺序调用(先编译、先调用)
b:调用子类的+load之前会先调用父类的+load
2:再调用分类的+load;
a:按照编译先后顺序调用(先编译,先调用)

三、Initialize

+initialize方法会在类第一次接收到消息时调用,走的是objc_sendMsg() (消息机制)。

调用顺序:
先调用父类的+initialize,再调用子类的+initialize方法(子类的可能不会调用)。
先初始化父类,后初始化子类,每个类只会初始化一次。

创建三个类,Person类,Student类,Teacher类,后两个继承自Person类。

如果父类实现了+initialize,而子类中没有实现,调用子类的子类的initialize方法,父类会被调用两次。


#import "Person.h"
 
@implementation Person
 
+ (void)initialize
{
    NSLog(@"person-initialize");
 
}
@end


#import "Person+Test.h"
 
@implementation Person (Test)
+ (void)initialize
{
    NSLog(@"person-test-initialize");
 
}
@end
 
#import "Student.h"
 
@implementation Student
@end

调用子类

// 调用
[Student alloc]

打印结果

2019-03-14 10:31:59.896 newxc[5535:1415054] person-test-initialize
2018-03-14 10:31:59.897 newxc[5535:1415054] person-test-initialize

Teacher也不实现方法,再调用Teacher类的

// 调用
[Student alloc];
[Teacher alloc];
// 结果
2019-03-14 10:38:35.784 newxc[5565:1419550] person-test-initialize
2019-03-14 10:38:35.785 newxc[5565:1419550] person-test-initialize
2019-03-14 10:38:35.786 newxc[5565:1419550] person-test-initialize

过程分析

a:看student类有没有初始化,没有->找到父类Person,没有初始化->初始化父类,打印第一次person-test-initalize。

b:调用父类的 继续往下走,会调用student的初始化方法,也就是发送objc_msgSend方法,通过isa发现Student类里面没有,通过superclass找到父类Person中发现有initalize方法,然后直接调用,注意:这个跟上面找initalize方法不一样,这个是直接调用,走的是普通类调用的正常流程,所以打印第二次person-test-initalize。

c:看Teacher有没有初始化,没有-> 找到父类Person,有初始化,略过,初始化自己,跟上面一样,跟objc_msgSend方法 通过isa 找到teacher中没有initalize方法,然后通过superclass找父类中有initalize方法,然后,直接调用。so,打印第三次person-test-initalize。

这里注意:父类的initalize方法调用了三次,不代表父类初始化了三次。第一次调用是在当时的类中 判断父类时调用的,后面的两次 是给student和teacher发消息时候调用的

说明:
initialize可以做一些懒加载,但是,initialize最大的缺陷是它是基于OC消息机制。
所以,如果子类没有实现initialize,那么会继续向父类发消息,直到找到为止。
因此,如果类A实现initialize,A的子类Aa没实现。假如A和Aa都被用到,A的initialize方法就会被调用2次。
通常这不是我们本意,常见的做法是在initialize方法里判断self是不是本类,若没有子类就不用了。

总结

想在类加载进内存的时候调用,就用load方法,如果想在第一次加载类的时候调用,就用initialized方法

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