可能有些还不清楚load和initialize的区别,下面简单说一下:

首先说一下 + initialize 方法:苹果官方对这个方法有这样的一段描述:这个方法会在 第一次初始化这个类之前 被调用,我们用它来初始化静态变量.

initialize方法的调用时机,当向该类发送第一个消息(一般是类消息首先调用,常见的是alloc)的时候,先调用类中的,再调用类别中的(类别中如果有重写);如果该类只是引用,没有调用,则不会执行initialize方法。
两者方法的共同点:自动调用父类的,不需要super操作;自动调用仅仅会调用一次(不包括外部显示调用).

load 方法会在加载类的时候就被调用,也就是 ios 应用启动的时候,就会加载所有的类,就会调用每个类的 + load 方法.

load方法的调用时机,main函数之前,先调用类中的,再调用类别中的(类别中如果有重写).

代码演示:

#pragram ---main函数中的代码---
#import <UIKit/UIKit.h>
#import "AppDelegate.h"
int main(int argc, char * argv[]) {
NSLog(@"%s",__func__);
@autoreleasepool {
return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
}
} #pragram ---基于NSObject的Person类---
#import "Person.h"
@implementation Person
+ (void)load{
NSLog(@"%s",__func__);
}
+ (void)initialize{
[super initialize];
NSLog(@"%s %@",__func__,[self class]);
}
- (instancetype)init{
if (self = [super init]) {
NSLog(@"%s",__func__);
}
return self;
}
@end #pragram ---基于Person的Son类---
#import "Girl.h"
@implementation Girl
+ (void)load{
NSLog(@"%s ",__func__);
}
+ (void)initialize{
[super initialize];
NSLog(@"%s ",__func__);
}
- (instancetype)init{
if (self = [super init]) {
NSLog(@"%s",__func__);
}
return self;
}
@end

输出日志:

-- ::36.535 initialize[:]] +[Person load]
-- ::36.535 initialize[:]] +[Girl load]
-- ::36.535 initialize[:]] main

这说明在我并没有对类做任何操作的情况下,+load 方法会被默认执行,并且是在 main 函数之前执行的。

#接下来我们来查看一下 + initialize 方法,先在 ViewController 中创建 Person 和 Girl 对象:

#import "ViewController.h"
#import "Person.h"
#import "Son.h"
#import "Girl.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
Person * p1 = [Person new];
Person * p2 = [Person new];
Girl *c1 = [Girl new];
Girl *c2 = [Girl new];
}
@end

输出日志:

-- ::57.134 initialize[:] +[Person load]
-- ::57.135 initialize[:] +[Girl load]
-- ::57.136 initialize[:] main
-- ::57.198 initialize[:] +[Person initialize] Person
-- ::57.198 initialize[:] -[Person init]
-- ::57.198 initialize[:] -[Person init]
-- ::57.198 initialize[:] +[Girl initialize]
-- ::57.199 initialize[:] -[Girl init]
-- ::57.199 initialize[:] -[Girl init]

+ initialize 方法类似一个懒加载,如果没有使用这个类,那么系统默认不会去调用这个方法,且默认只加载一次;

+ initialize 的调用发生在 +init 方法之前.

那么+ initialize 在父类与子类之间的关系是什么杨,我们创建一个继承自 Person 类的 Son类:

#pragram ---ViewController 中的代码---
#import "ViewController.h"
#import "Person.h"
#import "Son.h"
#import "Girl.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
Person * p1 = [Person new];
Person * p2 = [Person new];
Son*s = [Son new];
}
@end

输出日志:

-- ::14.140 initialize[:] +[Person load]
-- ::14.142 initialize[:] +[Son load]
-- ::14.142 initialize[:] +[Girl load]
-- ::14.142 initialize[:] main
-- ::14.203 initialize[:] +[Person initialize] Person
-- ::14.203 initialize[:] -[Person init]
-- ::14.203 initialize[:]] -[Person init]
-- ::14.204 initialize[:] +[Person initialize] Son
-- ::14.204 initialize[:] -[Person init]

我们会发现 Person 类的 + initialize 方法又被调用了,但是查看一下是子类 Son 调用的,也就是创建子类的时候,子类会去调用父类的 + initialize 方法。

这是因为在创建子类对象时,首先要创建父类对象,所以会调用一次父类的initialize方法,然后创建子类时,尽管自己没有实现initialize方法,但还是会调用到父类的方法。

虽然initialize方法对一个类而言只会调用一次,但这里由于出现了两个类,所以调用两次符合规则,但不符合我们的需求。正确使用initialize方法的姿势如下

// In Person.m
+ (void)initialize {
if (self == [Person class]) {
NSLog(@"Initialize Person, caller Class %@", [self class]);
}
}

加上判断后,就不会因为子类而调用到自己的initialize方法了.

总结:

  1. loadinitialize方法都会在实例化对象之前调用,以main函数为分水岭,前者在main函数之前调用,后者在之后调用。这两个方法会被自动调用,不能手动调用它们。
  2. loadinitialize方法都不用显示的调用父类的方法而是自动调用,即使子类没有initialize方法也会调用父类的方法,而load方法则不会调用父类。
  3. load方法通常用来进行Method Swizzle,initialize方法一般用于初始化全局变量或静态变量。
  4. loadinitialize方法内部使用了锁,因此它们是线程安全的。实现时要尽可能保持简单,避免阻塞线程,不要再使用锁。

最新文章

  1. JavaScript 智能社 完美运动框架
  2. 一个简单的excel文件上传到数据库方法
  3. sql(join on 和where的执行顺序)
  4. 【Stage3D学习笔记续】真正的3D世界(六):空间大战
  5. flash player over linux
  6. JSP技术
  7. scons小结
  8. Qt widget--杭州小笼包
  9. C# 操作Excel数据透视表
  10. .NETCore 下支持分表分库、读写分离的通用 Repository
  11. 【java】异常
  12. Javascrip错误类型
  13. FTP登录不上 显示“找不到元素”
  14. LOJ #6261 一个人的高三楼
  15. java实现点选汉字验证码(转)
  16. http Content-Type 知多少
  17. bootstrap4学习—Bootstrap v4.0.0-alpha.6的快速参考
  18. Rplidar学习(二)—— SDK库文件学习
  19. 20145307陈俊达《网络对抗》Exp2 后门原理与实践
  20. 【Chromium】sandboxed window问题记录

热门文章

  1. The record of Rf module debugging (1)
  2. 解析xml(当节点中有多个子节点)
  3. Django——ContentType(与多个表建立外键关系)及ContentType-signals的使用
  4. Cause: org.postgresql.util.PSQLException: ERROR: cached plan must not change result type的前因后果
  5. 三十分钟理解:双调排序Bitonic Sort,适合并行计算的排序算法
  6. Ubuntu文本编辑(vi和nano)命令
  7. 解决maven install报错信息(Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.1:compile )
  8. purescript 基本试用
  9. Apache Commons Codec 与消息摘要算法(hash算法)
  10. java scanner工具类