load和initialize方法

时间:2023-10-09 14:36:08

 

一、load

方法什么时候调用: 在main方法还没执行的时候 就会 加载所有类,调用所有类的load方法。

load方法是线程安全的,它使用了锁,我们应该避免线程阻塞在load方法。

在项目中使用的一些场景:

比如我们要统计所有页面(UIViewController、UITableViewController)的数据,可以在UIViewController的类别里的load方法里实现Method Swizzle

load和initialize方法
@implementation UIViewController (BaseMethod)

+ (void)load {
        Method originalViewDidLoad = class_getInstanceMethod([self class], @selector(viewDidLoad));
        Method swizzledViewDidLoad = class_getInstanceMethod([self class], @selector(swizzled_viewDidLoad));
        method_exchangeImplementations(originalViewDidLoad, swizzledViewDidLoad);
}
load和initialize方法

再比如AFNetworking中的_AFURLSessionTaskSwizzling类重写了load方法,并且在其中调用了swizzleResumeAndSuspendMethodForClass:来进行method swizzling。有兴趣的可以去看下源代码。

二、initialize

初始化对象的时候会调用initialize方法,initialize方法这块用来分配内存,init方法是创建对象。

具体到语法,就是 调用 alloc 方法的时候 会去执行initialize。

比如:  Bird *birdAlloc = [Bird alloc];  这句就会执行Bird类的initialize方法。

Bird *bird = [birdAlloc init];  这句会执行Bird的init方法。

initialize方法一般用于初始化全局变量或静态变量,也可以进行通知的处理。

在项目中的例子:

比如下面SBJson源代码中的代码,NSCharacterSet、NSMutableArray等无法在编译期初始化,可以放到initialize中进行赋值。

load和initialize方法
1 static NSCharacterSet *kDecimalDigitCharacterSet;
2
3 @implementation SBJsonTokeniser
4
5 + (void)initialize {
6     kDecimalDigitCharacterSet = [NSCharacterSet decimalDigitCharacterSet];
7 }
load和initialize方法

initialize内部也使用了锁,所以是线程安全的。但同时要避免阻塞线程,不要再使用锁。

与load方法类似的是,在initialize方法内部也会调用父类的方法,而且不需要我们显示的写出来。与load方法不同之处在于,即使子类没有实现initialize方法,也会调用父类的方法。

如果我们只需要父类的initialize方法只执行一次,可以这样写:

1 + (void)initialize {
2     if (self == [Bird class]) {
3         NSLog(@"%s %@", __func__, [self class]);
4     }
5 }

我们在init方法中会调用[super init],但load、initialize方法不需要我们调用super方法,调用super方法是多余的。

因为runtime会自动对父类load方法进行调用,而initialize则会随子类自动激发父类的方法,不需要显示调用。另一方面,如果父类中的方法用到的self(像示例中的方法),其指代的依然是类自身,而不是父类。

其他人总结的一个表格:

  +(void)load +(void)initialize
执行时机 在程序运行后立即执行 在类的方法第一次被调时执行
若自身未定义,是否沿用父类的方法?
类别中的定义 全都执行,但后于类中的方法 覆盖类中的方法,只执行一个

文章中涉及的代码放到 github 上了