2018-07-11 学习程序的执行顺序和UIViewController的生命周期

参考自这篇文章这篇对前文的整合

一.程序启动执行顺序

1.程序的入口

main函数,设置AppDelegate为函数的代理。

2.程序完成加载

-[AppDelegate application:didFinishLaunchingWithOptions:]

3.创建Window窗口

4.程序被激活

-[AppDelegate applicationDidBecomeActive:]

5.点击Home键

首先,程序会取消激活状态:-[AppDelegate applicationWillResignActive:]

然后,程序进入后台:-[AppDelegate applicationDidEnterBackground:]

6.重新点击进入程序

首先,程序进入前台:-[AppDelegate applicationWillEnterForeground:]

然后,程序会被激活:-[AppDelegate applicationDidBecomeActive:]

与Android的生命周期有着异曲同工之妙,但也有些许区别,比如创建进程之后,是加载\rightarrow激活的,没有applicationWillEnterForeground

  1. 创建进程

onCreate() / applicationDidFinishLaunching

  1. 进入视野(没获取焦点)

onStart() / applicationWillEnterForeground

  1. 获取焦点(可操作)

onResume() / applicationDidBecomeActive

  1. 失去焦点(还在视野内,不可操作)

onPause() / applicationWillResignActive

  1. 离开视野

onStop() / applicationDidEnterBackground

6.杀死程序

onDestroy() / applicationWillTerminate

2.在APPDelegate类实现的文件以及对应操作

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOption{// Override point for customization after application launch.
    NSLog(@"didFinishLaunchingWithOptions");
    return YES;
}

/* 当应用程序从活动状态(active)变到非活动状态(inactive时被触发调用, 这可能发生在一些临时中断下(例如:来电话、来短信)又或者程序退出时,他会先过渡到后台然后terminate 使用这方法去暂停正在进行的任务,禁用计时器,节流OpenGL ES 帧率。在游戏中应该在这个方法里面暂停游戏。 */
- (void)applicationWillResignActive:(UIApplication *)application { 
    // Sent when the application is about to move from active to inactive state. This can occur for certain types of temporary interruptions (such as an incoming phone call or SMS message) or when the user quits the application and it begins the transition to the background state.    // Use this method to pause ongoing tasks, disable timers, and throttle down OpenGL ES frame rates. Games should use this method to pause the game.
    NSLog(@"WillResignActive");
}

 /* 使用这种方法来释放共享资源,保存用户数据,无效计时器,存储足够多的应用程序状态信息来恢复您的应用程序的当前状态,以防它终止丢失数据。 如果你的程序支持后台运行,那么当用户退出时不会调用applicationWillTerminate。 */
- (void)applicationDidEnterBackground:(UIApplication *)application {
    // Use this method to release shared resources, save user data, invalidate timers, and store enough application state information to restore your application to its current state in case it is terminated later.      // If your application supports background execution, this method is called instead of applicationWillTerminate: when the user quits.
        NSLog(@"DidEnterBackground");
    }

 /* 先从后台切换到非活动状态,然后进入活动状态。 */
- (void)applicationWillEnterForeground:(UIApplication *)application {
    // Called as part of the transition from the background to the inactive state; here you can undo many of the changes made on entering the background.
    NSLog(@"WillEnterForeground");
}

/* 重启所有的任务,不管是从非活动状态还是刚启动程序,还是后台状态。 */
- (void)applicationDidBecomeActive:(UIApplication *)application { 
    // Restart any tasks that were paused (or not yet started) while the application was inactive. If the application was previously in the background, optionally refresh the user interface.
        NSLog(@"DidBecomeActive");
    }

/* 终止,game over */
- (void)applicationWillTerminate:(UIApplication *)application { 
    // Called when the application is about to terminate. Save data if appropriate. See also applicationDidEnterBackground:.
    NSLog(@"WillTerminate");
}

刚启动时出现了两个:


启动

按下Home键又出现了两个:


后台

点回去又出现了两个:


回到前台

由此,可以得到以下生命周期图:

活动和非活动

生命周期图

对于每个阶段的分析:

  • application: didFinishLaunchingWithOptions:程序首次已经完成启动时执行,一般在这个函数里面创建Window对象,将程序内容通过Window呈现给用户。

  • applicationWillResignActive:程序将要失去Active状态时调用,比如有电话进来或者按下Home键,之后程序进入后台状态,对应applicationWillEnterForeground方法。该函数主要执行的操作有:
    a. 暂停正在执行的任务
    b. 禁止计时器
    c. 减少OpenGL ES帧率
    d. 若为游戏则暂停游戏

  • applicationDidEnterBackground(已经进入后台):对应的是applicationDidBecomeActive(已经变成前台)。该方法用来:
    a. 释放共享资源
    b. 保存用户数据(写到硬盘)
    c. 作废计时器
    d. 保存足够的程序状态以便下次修复

  • applicationWillEnterForeground(即将进入前台):用来撤销applicationWillResignActive(即将进入后台)做出的改变。

  • applicationDidBecomeActive(已经进入前台):程序已经变为Active时调用,对应applicationDidEnterBackground。若程序之前在后台,则在此方法内刷新用户界面。

  • applicationWillTerminate:程序即将退出时调用,记得保存数据,如applicationDidEnterBackground一样。

二.UIViewController生命周期

之前在视频器播放项目中遇到过关于ViewController的问题反馈,现在再把它拿出来:

首先看看单个viewController的生命周期:

  1. loadView:加载view 会多次调用并且会使viewWillLayoutSubviewsviewDidLayoutSubviews不再执行
  2. viewDidLoadview加载完毕
  3. viewWillAppear:控制器的view将要显示
  4. viewWillLayoutSubviews:控制器的view将要布局子控件
  5. viewDidLayoutSubviews:控制器的view布局子控件完成
    这期间系统可能会多次调用viewWillLayoutSubviewsviewDidLayoutSubviews 两个方法
  6. viewDidAppear:控制器的view完全显示
  7. viewWillDisappear:控制器的view即将消失的时候
  8. viewDidDisappear:控制器的view完全消失的时候

整个控制器生命周期:loadView-> viewDidLoad -> viewWillAppear -> viewWillLayoutSubviews -> viewDidLayoutSubviews -> viewDidAppear -> viewWillDisappear -> viewDidDisappear

  • viewDidLoad:当程序第一次加载view时调用,以后都不会用到,而viewDidAppear是每当切换到view时就调用。

  • viewWillAppear:系统在载入所有的数据之后,将会在屏幕上显示视图,这时候会先调用这个方法,通常我们会在这个方法对即将显示的视图做进一步的设置,如:设置设备不同方向的时候该如何显示;设置状态栏方向、设置视图显示样式等等。

  • viewWillLayoutSubviews :在 viewWillAppear 之后, viewDidAppear 之前执行,view即将布局其subViews,比如viewbounds发生了改变(状态栏从不显示到显示、视图方向发生变化),在调整之前要做的工作可以放在这个方法实现。

注意:这个方法会被调用多次,如果在此创建视图,可能会创建多个,而且这个方法中执行耗时操作依然会造成跳转卡顿的问题。

  • viewDidLayoutSubviews:放置调整之后的工作

  • didReceiveMemoryWarning:在内存足够的情况下,APP的视图通常会一直保存在内存中,但如果内存不够,一些没有正在显示的View Controller就会收到内存不足的警告,然后就会释放自己拥有的视图,以达到释放内存的目的。但是系统只会释放内存,并不会释放对象的所有权,所以通常将不需要显示在内存中保留的对象释放它的所有权,将其指针置为nil

下面写个程序来试试:

//视图控制器中的视图加载完成,viewController自带的view加载完成
- (void)viewDidLoad {
    NSLog(@"One--%s", __FUNCTION__);
    [super viewDidLoad];
    
    UIButton *button = [[UIButton alloc] initWithFrame: CGRectMake(self.view.frame.size.width / 2 - 100,
                                                                   self.view.frame.size.height / 2 - 30,
                                                                   100,
                                                                   30)];
    [self.view addSubview: button];
    [button setTitle: @"Click" forState: UIControlStateNormal];
    [button setBackgroundColor: [UIColor blueColor]];
    [button addTarget: self action: @selector(buttonClicked:) forControlEvents: UIControlEventTouchUpInside];
    
    
    // Do any additional setup after loading the view.
}

- (void) buttonClicked: (UIButton *) sender {
    AnotherViewController *anotherViewController = [[AnotherViewController alloc] init];
    [self.navigationController pushViewController: anotherViewController animated: YES];
    
}
//出现内存警告  //模拟内存警告:点击模拟器->hardware-> Simulate Memory Warning
- (void)didReceiveMemoryWarning {
    NSLog(@"One--%s", __FUNCTION__);
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
}

//视图将要出现
- (void)viewWillAppear:(BOOL)animated {
    NSLog(@"One--%s", __FUNCTION__);
    [super viewWillAppear:animated];
}

//将要布局子视图
- (void) viewWillLayoutSubviews {
    NSLog(@"One--%s", __FUNCTION__);
    [super viewWillLayoutSubviews];
}

//布局子视图完毕
- (void)viewDidLayoutSubviews {
    NSLog(@"One--%s", __FUNCTION__);
    [super viewDidLayoutSubviews];
}

//视图已经出现
- (void)viewDidAppear:(BOOL)animated {
    NSLog(@"One--%s", __FUNCTION__);
    [super viewDidAppear:animated];
}

//视图将要消失 //双击Home键,向上推出程序执行该函数
- (void)viewWillDisappear:(BOOL)animated {
    NSLog(@"One--%s", __FUNCTION__);
    [super viewWillDisappear:animated];
}

//视图已经消失
- (void)viewDidDisappear:(BOOL)animated {
    NSLog(@"One--%s", __FUNCTION__);
    [super viewDidDisappear:animated];
}

下面来演示一下各种情况各个步骤的执行顺序:

1.启动

viewDidLoad -> viewWillAppear -> viewWillLayoutSubviews -> viewDidLayoutSubviews -> viewDidAppear

2.模拟Memory Warning

点击模拟器->hardware-> Simulate Memory

3.跳转到Controller2

可以看到,两个View Controller的生命周期是很符合我们设想的逻辑的,2加载完毕后告知1准备消失,2准备出现,然后2可以进行子布局的布置,布置完毕后,1先消失,随后2出现。

2--viewDidLoad -> 1--viewWillDisappear -> 2--viewWillAppear -> 2--viewWillLayoutSubviews -> 2--viewDidLayoutSubviews -> 1--viewDidDisappear -> 2--viewDidAppear

4.Controller2出栈

出栈的过程其实就是免去了加载布局和布置子布局,因此他的逻辑顺序同样是消失的先执行,出现的后执行,如下所示:

2--viewWillDisappear -> 1--viewWillAppear -> 2--viewDidDisappear -> 1--viewDidAppear

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,186评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,858评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,620评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,888评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,009评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,149评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,204评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,956评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,385评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,698评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,863评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,544评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,185评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,899评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,141评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,684评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,750评论 2 351