Objective-C 有两个神奇的方法:+load 和 +initialize,这两个方法在类被使用时会自动调用。但是两个方法的不同点会导致应用层面上性能的显著差异。

一、+ initialize 方法和+load 调用时机

  • 首先说一下 + initialize 方法:苹果官方对这个方法有这样的一段描述:这个方法会在 第一次初始化这个类之前 被调用,我们用它来初始化静态变量。
    • load 方法会在加载类的时候就被调用,也就是 ios 应用启动的时候,就会加载所有的类,就会调用每个类的 + load 方法。
  • 之后我们结合代码来探究一下 + initialize 与 + load 两个方法的调用时机,首先是 + load
    1. #pragram ---main函数中的代码---
    1. #import <UIKit/UIKit.h>
    2. #import "AppDelegate.h"
    3. int main(int argc, char * argv[]) {
    4. NSLog(@"%s",__func__);
    5. @autoreleasepool {
    6. return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
    7. }
    8. }
    1. #pragram ---基于NSObject的Person类---
    1. #import "Person.h"
    2. @implementation Person
    3. + (void)load{ NSLog(@"%s",__func__); }
    4. + (void)initialize{
    5. [super initialize];
    6. NSLog(@"%s %@",__func__,[self class]);
    7. }
    8. - (instancetype)init{
    9. if (self = [super init]){
    10. NSLog(@"%s",__func__);
    11. }
    12. return self;
    13. }
    14. @end
    1. #pragram ---基于Person的Son类---
    1. #import "Girl.h"
    2. @implementation Girl
    3. + (void)load{
    4. NSLog(@"%s ",__func__);
    5. }
    6. + (void)initialize{
    7. [super initialize];
    8. NSLog(@"%s ",__func__);
    9. }
    10. - (instancetype)init{
    11. if (self = [super init]){
    12. NSLog(@"%s",__func__);
    13. }
    14. return self;
    15. }
    16. @end

    运行程序,我们看一下输出日志:

    1. 2015-10-27 15:21:07.545 initialize[11637:334237] +[Person load]
    2. 2015-10-27 15:21:07.546 initialize[11637:334237] +[Girl load]
    3. 2015-10-27 15:21:07.546 initialize[11637:334237] main

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

  • 接下来我们来查看一下 + initialize 方法,先在 ViewController 中创建 Person 和 Girl 对象:
    1. #import "ViewController.h"
    2. #import "Person.h"
    3. #import "Son.h"
    4. #import "Girl.h"
    5. @interface ViewController ()
    6. @end
    7. @implementation ViewController
    8. - (void)viewDidLoad {
    9. [super viewDidLoad];
    10. Person * a = [Person new];
    11. Person * b = [Person new];
    12. Girl *c = [Girl new];
    13. Girl *d = [Girl new];
    14. }
    15. @end

    下面我们来看一下输出日志:

    1. 2015-10-27 15:33:56.195 initialize[11711:342410] +[Person load]
    2. 2015-10-27 15:33:56.196 initialize[11711:342410] +[Girl load]
    3. 2015-10-27 15:33:56.197 initialize[11711:342410] main
    4. 2015-10-27 15:33:56.259 initialize[11711:342410] +[Person initialize] Person
    5. 2015-10-27 15:33:56.259 initialize[11711:342410] -[Person init]
    6. 2015-10-27 15:33:56.259 initialize[11711:342410] -[Person init]
    7. 2015-10-27 15:33:56.259 initialize[11711:342410] +[Girl initialize]
    8. 2015-10-27 15:33:56.260 initialize[11711:342410] -[Girl init]
    9. 2015-10-27 15:33:56.260 initialize[11711:342410] -[Girl init]

    通过这个实验我们可以确定两点:

    • + initialize 方法类似一个懒加载,如果没有使用这个类,那么系统默认不会去调用这个方法,且默认只加载一次;
    • + initialize 的调用发生在 +init 方法之前。
  • 接下来再探究一下 + initialize 在父类与子类之间的关系,创建一个继承自 Person 类的 Son类:
    1. #pragram ---ViewController 中的代码---
    1. #import "ViewController.h"
    2. #import "Person.h"
    3. #import "Son.h"
    4. #import "Girl.h"
    5. @interface ViewController ()
    6. @end
    7. @implementation ViewController
    8. - (void)viewDidLoad {
    9. [super viewDidLoad];
    10. Person * a = [Person new];
    11. Person * b = [Person new];
    12. Son*z = [Son new];
    13. }
    14. @end

    看一下输出日志:

    1. 2015-10-27 15:44:55.762 initialize[12024:351576] +[Person load]
    2. 2015-10-27 15:44:55.764 initialize[12024:351576] +[Son load]
    3. 2015-10-27 15:44:55.764 initialize[12024:351576] +[Girl load]
    4. 2015-10-27 15:44:55.764 initialize[12024:351576] main
    5. 2015-10-27 15:44:55.825 initialize[12024:351576] +[Person initialize] Person
    6. 2015-10-27 15:44:55.825 initialize[12024:351576] -[Person init]
    7. 2015-10-27 15:44:55.825 initialize[12024:351576] -[Person init]
    8. 2015-10-27 15:44:55.826 initialize[12024:351576] +[Person initialize] Son
    9. 2015-10-27 15:44:55.826 initialize[12024:351576] -[Person init]

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


二,使用场景

+load():通常用来进行Method Swizzle,尽量避免过于复杂以及不必要的代码
示例:

  1. + (void)load {
  2. Method originalFunc = class_getInstanceMethod([self class], @selector(originalFunc));
  3. Method swizzledFunc = class_getInstanceMethod([self class], @selector(swizzledFunc));
  4. method_exchangeImplementations(originalFunc, swizzledFunc);
  5. }

+initialize():一般用于初始化全局变量或静态变量
示例:

  1. #import "SubscriptionServiceCenter.h"
  2.  
  3. static NSMutableDictionary *_subscriptionDictionary = nil;
  1. @implementation SubscriptionServiceCenter
  2. + (void)initialize {
  3. if (self == [SubscriptionServiceCenter class]) {
  4. _subscriptionDictionary = [NSMutableDictionary dictionary];
  5. }
  6. }

三、总结

  • 如果你实现了 + load 方法,那么当类被加载时它会自动被调用。这个调用非常早。如果你实现了一个应用或框架的 + load,并且你的应用链接到这个框架上了,那么 + load 会在 main() 函数之前被调用。如果你在一个可加载的 bundle 中实现了 + load,那么它会在 bundle 加载的过程中被调用。
  • + initialize 方法的调用看起来会更合理,通常在它里面写代码比在 + load 里写更好。+ initialize 很有趣,因为它是懒调用的,也有可能完全不被调用。类第一次被加载时,
  • + initialize 不会被调用。类接收消息时,运行时会先检查 + initialize 有没有被调用过。如果没有,会在消息被处理前调用。

iOS-方法之+ initialize 与 +load的更多相关文章

  1. iOS Load方法 和 initialize方法的比较

    一.load方法特点: 1. 当类被引用进程序的时候会执行这个函数 2.一个类的load方法不用写明[super load],父类就会收到调用,并且在子类之前. 3.Category的load也会收到 ...

  2. iOS - + initialize 与 +load

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

  3. iOS之initialize与load

    initialize和load 这两个方法都是是什么时候调用的呢?都有着什么样的作用,下面看看吧! initialize +(void)initialize{ } 什么时候调用:当第一次使用这个类的时 ...

  4. 【OC底层】Category、+load方法、+initialize方法原理

    Category原理 - Category编译之后的底层结构是 struct categroy_t,里面存储着分类对象方法.属性.协议信息- 当程序运行时,通过runtime动态的将分类的方法.属性. ...

  5. Objective-C中的+initialize和+load

    写在前面 近几天花了一些时间了解了一下Objective-C runtime相关的东西,其中涉及到了+load方法,譬如method swizzling通常在category的+load方法中完成.之 ...

  6. iOS方法类:CGAffineTransform的使用大概

    CoreGraphics框架中的CGAffineTransform类可用于设定UIView的transform属性,控制视图的缩放.旋转和平移操作: 另称放射变换矩阵,可参照线性代数的矩阵实现方式0. ...

  7. [原创]java WEB学习笔记79:Hibernate学习之路--- 四种对象的状态,session核心方法:save()方法,persist()方法,get() 和 load() 方法,update()方法,saveOrUpdate() 方法,merge() 方法,delete() 方法,evict(),hibernate 调用存储过程,hibernate 与 触发器协同工作

    本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ...

  8. iOS方法类:CGAffineTransform的使用

    CoreGraphics框架中的CGAffineTransform类可用于设定UIView的transform属性,控制视图的缩放.旋转和平移操作: 另称放射变换矩阵,可参照线性代数的矩阵实现方式0. ...

  9. iOS同步后couldn't load project

    使用Cornerstone同步后,有时候出现下面的现象,点击运行,提示“couldn't load project” 解决方法: 1.鼠标右键点击工程“XX_Internal”,选择“show in ...

随机推荐

  1. 第三部分:Android 应用程序接口指南---第一节:应用程序组件---第一章1-1.Fragment

    第1-1章 Fragments 在Activity中的fragment代表的是一种行为或用户界面的一部分.你可以在activity中结合多个fragments创建一个多面板UI,并可以在多个activ ...

  2. flutter 环境安装以及配置

    首先需要下载flutter源码,以下是github地址: https://github.com/flutter/flutter 然后需要安装git环境吧,下图红框可以自行下载安装 接下来需要安装flu ...

  3. Netflix开源类库archaius(一)概述

    archaius是什么,能做什么? archaius是Netflix公司开源项目之一,基于java的配置管理类库,主要用于多配置存储的动态获取.主要功能是对apache common configur ...

  4. ELK+Filebeat+Kafka+ZooKeeper 构建海量日志分析平台

    日志分析平台,架构图如下: 架构解读 : (整个架构从左到右,总共分为5层) 第一层.数据采集层 最左边的是业务服务器集群,上面安装了filebeat做日志采集,同时把采集的日志分别发送给两个logs ...

  5. windows下添加多个git仓库账号

    当使用git方式下载时,如果没有配置过ssh key,会提示错误(git clone支持https和git(即ssh)两种方式下载源码) 当需要在机器上使用不同的git账户,这就需要知道如何在机器上添 ...

  6. 九度OJ 1067 n的阶乘 (模拟)

    题目1067:n的阶乘 时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:5666 解决:2141 题目描写叙述: 输入一个整数n,输出n的阶乘 输入: 一个整数n(1<=n<=2 ...

  7. C#基础系列——语法

    1.C#是这样开始的: 函数入口:static void  Main(String [] args){} 2.Hello  World 例子 using system;--------导入命名空间,里 ...

  8. Gradle 设置全局代理

    #systemProp.socks.proxyHost=127.0.0.1 #systemProp.socks.proxyPort=7077 #systemProp.https.proxyHost=1 ...

  9. css3整理--Animation

    animation语法: 1.动画的定义 @keyframes IDENT { from { Properties:Properties value; } Percentage { Propertie ...

  10. iOS - 导航栏设置半透明或取消半透明

    self.navigationController.navigationBar.translucent = YES;//透明