浅析UIApplication生命周期的一些delegate方法

前言

网上有很多关于程序启动过程和UIApplicationDelegate方法调用顺序的文章。笔者这里不再介绍程序的启动过程和delegate方法的调用过程。而是介绍一下UIApplication会在什么情况下调用UIApplicationDelegate的哪些方法。以及常见的场景下,哪些方法会被调用,苹果为什么会这样做。

回顾

首先让我们先来回顾下与程序启动过程相关的一些delegate方法的调用时机。

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  NSLog(@"程序启动完成:%s",__func__);
  return YES;
}

- (void)applicationDidBecomeActive:(UIApplication *)application {
    NSLog(@"已经获得焦点:%s",__func__);
}

- (void)applicationWillResignActive:(UIApplication *)application {
    NSLog(@"将要释放焦点:%s",__func__);
}

- (void)applicationDidEnterBackground:(UIApplication *)application {
    NSLog(@"已经进入后台:%s",__func__);
}

- (void)applicationWillEnterForeground:(UIApplication *)application {
    NSLog(@"将要进入前台:%s",__func__);
}

- (void)applicationWillTerminate:(UIApplication *)application {
    NSLog(@"程序将要退出:%s",__func__);
}

情景一 程序启动

程序被加载到内存,完成启动,application对象会自动调用delegate的下面这个方法,证明程序已经启动完成。所以这个方法也是首先会被application回调的方法,且这个方法在整个程序的生命周期中只会被调用一次。

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions;

程序启动时,回调完上面的方法,会继续回调delegate的已经获得了焦点的方法,证明程序已经获得了焦点

- (void)applicationDidBecomeActive:(UIApplication *)application;
```

>结论:应用启动过程中,会依次调用delegate已经启动完成和已经获得焦点的方法,不会调用已经进入前台的方法。

#### 情景二  程序从前台退出到后台
当程序处于前台时,单击home键,程序会自动退出到后台。在这个过程中,程序会先回调delegate的将要失去焦点的方法,证明程序**`将要失去焦点`**。
```
- (void)applicationWillResignActive:(UIApplication *)application;
```

调用调用完上面的方法后,程序紧接着会调用delegate已经进入后台的方法,证明**`程序已经进入后台`**。
```
- (void)applicationDidEnterBackground:(UIApplication *)application;
```

> 结论:单击home键进入后台会依次调用delegate的将要失去焦点的方法和已经进入后台的方法。

####情景三  程序从后台进入到前台
(4)从后台进入前台(无论是双击home键进入或者点击应用图标进入),会回调delegate的将要进入前台方法,证明**`程序将要进入前台`**。
```
- (void)applicationWillEnterForeground:(UIApplication *)application;
```

回调完上面的方法,紧接着会继续回调delegate的已经获得焦点的方法,证明程序**`已经获得了焦点`**。
```
- (void)applicationDidBecomeActive:(UIApplication *)application;
```

> 结论:从后台进入前台,会依次调用delegate的将要进入前台和已经获得焦点的方法。


#### 情景四  双击home键切换程序
在前台,双击home键,只会调用delegate的将要失去焦点的方法,证明```程序将要失去焦点```。
```
- (void)applicationWillResignActive:(UIApplication *)application;
```


当用户真正切换应用时候,才会继续调用delegate的已经进入后台的方法,证明**`程序已经进入后台`**。

```
- (void)applicationDidEnterBackground:(UIApplication *)application;
```

> 结论:双击home键切换应用。会分别调用程序将要失去焦点的方法和程序已经进入后台的方法。 且这两个方法是分开调用的。即,双击home键时调用将要失去焦点的方法,选择其他应用时调用已经进入后台的方法。

####情景五 在前台双击home键杀死程序

双击home键时,只会调用delegate的将要失去焦点的方法(上面已经说过),证明程序将要失去焦点。
```
- (void)applicationWillResignActive:(UIApplication *)application;
```


然后手指上滑杀死程序,会直接调用delegate的已经进入后台的方法,证明程序已经进入后台。
```
- (void)applicationDidEnterBackground:(UIApplication *)application;
```
然后紧接着调用delegate的程序将要退出的方法,证明程序将要被杀死。
- (void)applicationWillTerminate:(UIApplication *)application;

> 结论:双击home键然后杀死程序,会按照如下顺序调用delegate的方法:
```
- (void)applicationWillResignActive:(UIApplication *)application;(双击home键调用)
```
```
- (void)applicationDidEnterBackground:(UIApplication *)application;(杀死程序时调用这两个方法)
- (void)applicationWillTerminate:(UIApplication *)application;
```

####情景六 从其他程序前台双击home键杀死后台程序

如果从其他程序的前台,双击home键杀死后台程序,被杀死程序只会回调delegate即将退出的方法。
```
- (void)applicationWillTerminate:(UIApplication *)application;
```

**为什么呢?**
因为我们是从一个前台程序杀死一个后台程序,这个后台程序当初进入后台时候已经调用了将要释放焦点和已经进入后台的方法,所以杀死时候只会回调delegate即将终结的方法。

> 结论:从一个前台程序杀死一个后台程序。后台程序只会回调delegate的程序即将退出的方法。

---

####情景七 下拉通知栏
下拉通知栏,只会回调delegate的程序将要释放焦点的方法。程序并没有进入后台,所以不会调用进入后台的方法
```
- (void)applicationWillResignActive:(UIApplication *)application;
```

结论:下拉状态栏只会让程序失去焦点,并不会让程序进入后台。

因为下拉通知栏只调用了将要释放焦点的方法,没有调用进入后台方法,所以收起通知栏时,只会调用已经获得焦点的方法,不会调用进入前台的方法。
```
- (void)applicationDidBecomeActive:(UIApplication *)application;
```

同样,从屏幕下方向上滑动屏幕,唤出工具栏时候,也只会调用delegate的将要释放焦点的方法。收起工具栏时,只会调用delegate的已经获得焦点的方法。

> 结论:下拉通知栏或者上拉工具栏,都只是回调delegate的即将释放焦点的方法,程序不会进入后台。

## 为什么
当初学习iOS时候,对这个地方不是很清楚,总是搞不懂为什么程序的delegate有一个将要进入前台的方法`applicationWillEnterForeground:`,却没有类似于`applicationDidEnterForeground:`的已经进入前台的方法(纯属捏造)?为什么程序的delegate有一个已经进入后台的方法`applicationDidEnterBackground:`却没有一个类似于`applicationWillEnterBackground:`的将要进入后台的方法?为什么进入前台时,方法的调用顺序是`applicationWillEnterForeground:`和`applicationDidBecomeActive:`而不是相反?这些问题一直困扰着我。

**将要进入前台、已经获得焦点、将要失去焦点、已经进入后台**这几个方法是比较容易混淆的,且调用顺序经常被搞混。但是如果理解了苹果为什么这么设计,这些困惑都将迎刃而解。重点来了:如果一个应用程序失去焦点那么意味着用户当前无法进行交互操作,正因如此,程序从前台退出到后台时候,一般会**先失去焦点再进入后台**避免进入后台过程中用户还可以和程序进行交互。同理,一个应用程序从后台进入前台也是类似的,会**先进入前台再获得焦点**,这样进入前台过程中未完全准备好的情况下用户无法操作,保证了程序的安全性。
至于为什么苹果没有提供类似于`applicationDidEnterForeground:`的已经进入前台的方法,那是因为程序进入前台后必定会回调delegate的已经获得焦点的方法,所以`applicationDidBecomeActive:`方法从本质上就相当于我们想象中的`applicationDidEnterForeground:`,如果我们想要在程序进入前台后做什么操作,完全可以把这些操作写到`applicationDidBecomeActive:`里。同理,`applicationWillResignActive:`就相当于我们想象中的`applicationWillEnterForeground:`。

另外一般如果**应用程序要保存用户数据会在程序将要失去焦点的方法中进行 (而不是在已经进入后台的方法中执行)**,因为如果用户双击Home不会进入后台只会注销激活。同理,如果用户恢复应用状态一般在已经获的焦点的方法中执行(而不是在将要进入前台的方法中执行)。

  文/VV木公子(简书作者)
**PS:如非特别说明,所有文章均为原创作品,著作权归作者所有,转载请联系作者获得授权,并注明出处,所有打赏均归本人所有!**

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

推荐阅读更多精彩内容