一、load
方法什么时候调用: 在main方法还没执行的时候 就会 加载所有类,调用所有类的load方法。
load
方法是线程安全的,它使用了锁,我们应该避免线程阻塞在load
方法。
在项目中使用的一些场景:
比如我们要统计所有页面(UIViewController、UITableViewController)的数据,可以在UIViewController的类别里的load方法里实现Method Swizzle
@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); }
再比如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中进行赋值。
1 static NSCharacterSet *kDecimalDigitCharacterSet; 2 3 @implementation SBJsonTokeniser 4 5 + (void)initialize { 6 kDecimalDigitCharacterSet = [NSCharacterSet decimalDigitCharacterSet]; 7 }
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 上了