一、+ initialize 方法和+load 调用时机
首先说一下 + initialize 方法:苹果官方对这个方法有这样的一段描述:这个方法会在第一次初始化这个类之前被调用,我们用它来初始化静态变量。
load 方法会在加载类的时候就被调用,也就是 ios 应用启动的时候,就会加载所有的类,就会调用每个类的 + load 方法。
之后我们结合代码来探究一下 + initialize 与 + load 两个方法的调用时机,首先是+ load:
#pragram ---main函数中的代码---#import#import"AppDelegate.h"intmain(intargc,char* argv[]) {NSLog(@"%s",__func__);@autoreleasepool{returnUIApplicationMain(argc, argv,nil,NSStringFromClass([AppDelegateclass])); }}#pragram ---基于NSObject的Person类---#import"Person.h"@implementationPerson+ (void)load{NSLog(@"%s",__func__);}+ (void)initialize{ [superinitialize];NSLog(@"%s %@",__func__,[selfclass]);}- (instancetype)init{if(self= [superinit]) {NSLog(@"%s",__func__); }returnself;}@end#pragram ---基于Person的Son类---#import"Girl.h"@implementationGirl+ (void)load{NSLog(@"%s ",__func__);}+ (void)initialize{ [superinitialize];NSLog(@"%s ",__func__);}- (instancetype)init{if(self= [superinit]) {NSLog(@"%s",__func__); }returnself;}@end
运行程序,我们看一下输出日志:
2015-10-2715:21:07.545initialize[11637:334237]+[Person load]2015-10-2715:21:07.546initialize[11637:334237]+[Girl load]2015-10-2715:21:07.546initialize[11637:334237]main
这说明在我并没有对类做任何操作的情况下,+load 方法会被默认执行,并且是在 main 函数之前执行的。
接下来我们来查看一下+ initialize方法,先在 ViewController 中创建 Person 和 Girl 对象:
#import"ViewController.h"#import"Person.h"#import"Son.h"#import"Girl.h"@interfaceViewController ()@end@implementationViewController- (void)viewDidLoad { [superviewDidLoad]; Person * a = [Personnew]; Person * b = [Personnew]; Girl *c = [Girlnew]; Girl *d = [Girlnew];}@end
下面我们来看一下输出日志:
2015-10-2715:33:56.195initialize[11711:342410]+[Person load]2015-10-2715:33:56.196initialize[11711:342410]+[Girl load]2015-10-2715:33:56.197initialize[11711:342410]main2015-10-2715:33:56.259initialize[11711:342410]+[Person initialize]Person2015-10-2715:33:56.259initialize[11711:342410]-[Person init]2015-10-2715:33:56.259initialize[11711:342410]-[Person init]2015-10-2715:33:56.259initialize[11711:342410]+[Girl initialize]2015-10-2715:33:56.260initialize[11711:342410]-[Girl init]2015-10-2715:33:56.260initialize[11711:342410]-[Girl init]
通过这个实验我们可以确定两点:
+ initialize方法类似一个懒加载,如果没有使用这个类,那么系统默认不会去调用这个方法,且默认只加载一次;
+ initialize的调用发生在 +init 方法之前。
接下来再探究一下+ initialize在父类与子类之间的关系,创建一个继承自 Person 类的 Son类:
#pragram ---ViewController 中的代码---#import"ViewController.h"#import"Person.h"#import"Son.h"#import"Girl.h"@interfaceViewController ()@end@implementationViewController- (void)viewDidLoad { [superviewDidLoad]; Person * a = [Personnew]; Person * b = [Personnew]; Son*z = [Sonnew];}@end
看一下输出日志:
2015-10-2715:44:55.762initialize[12024:351576]+[Person load]2015-10-2715:44:55.764initialize[12024:351576]+[Son load]2015-10-2715:44:55.764initialize[12024:351576]+[Girl load]2015-10-2715:44:55.764initialize[12024:351576]main2015-10-2715:44:55.825initialize[12024:351576]+[Person initialize]Person2015-10-2715:44:55.825initialize[12024:351576]-[Person init]2015-10-2715:44:55.825initialize[12024:351576]-[Person init]2015-10-2715:44:55.826initialize[12024:351576]+[Person initialize]Son2015-10-2715:44:55.826initialize[12024:351576]-[Person init]
我们会发现 Person 类的+ initialize方法又被调用了,但是查看一下是子类 Son 调用的,也就是创建子类的时候,子类会去调用父类的+ initialize方法。
二、总结
如果你实现了+ load方法,那么当类被加载时它会自动被调用。这个调用非常早。如果你实现了一个应用或框架的+ load,并且你的应用链接到这个框架上了,那么+ load会在 main() 函数之前被调用。如果你在一个可加载的 bundle 中实现了+ load,那么它会在 bundle 加载的过程中被调用。
+ initialize方法的调用看起来会更合理,通常在它里面写代码比在+ load里写更好。+ initialize很有趣,因为它是懒调用的,也有可能完全不被调用。类第一次被加载时,
+ initialize不会被调用。类接收消息时,运行时会先检查+ initialize有没有被调用过。如果没有,会在消息被处理前调用。
补充:
1.load方法的调用时机,main函数之前,先调用类中的,再调用类别中的(类别中如果有重写);
2.initialize方法的调用时机,当向该类发送第一个消息(一般是类消息首先调用,常见的是alloc)的时候,先调用类中的,再调用类别中的(类别中如果有重写);如果该类只是引用,没有调用,则不会执行initialize方法。
两者方法的共同点:自动调用父类的,不需要super操作;自动调用仅仅会调用一次(不包括外部显示调用)。