iOS应用程序的状态及其切换(生命周期)

0. iOS应用入口和AppDelegate

我最初开始iOS应用开发学习的时候,也没有像样的培训和指导,第一个要看的就是代码。我们知道Objective-C也是基于C的,于是我们找到了入口代码main()函数。通常代码如下:

1
2
3
4
5
6
int main(int argc, char *argv[])
{
   @autoreleasepool {
      return UIApplicationMain(argc, argv, nil, NSStringFromClass([MyAppDelegate class]));
   }
}

我们看到返回类型、函数名和参数都很熟悉。至于函数体,除了AutoRelease是自动释放池之外,执行的是一个名为UIApplicationMain的函数调用,后两个参数是nil(空)和一个MyAppDelegate类名字符串。

这个函数调用也就是这个iOS应用的开始,它进行了这个应用的初始化过程,并生成了一个类名为MyAppDelegate的对象。至于MyAppDelegate这个名字,这里只是一个示例,可以为其它命名,通常是应用统一定义的前缀+”AppDelegate”。这些都不重要,重要的是它实际上需要遵从于UIApplicationDelegate这个Protocol的定义,给出一些方法的实现。

那么这个“AppDelegate”又是什么呢?实际上,苹果已经为把每个应用包装成一个UIApplication对象,但应用每一步运行的细节并不需要开发者关注,只要关注这个应用对象对应的delegate即可,也就是这个“AppDelegate”。通过AppDelegate我们可以知道Application的运行状态,发生了哪些事件。

1. 程序的5个状态和对应的AppDelegate的7个方法

至于五个状态,分别是:

  • Not Running, 未运行
  • Inactive,  非活动
  • Active, 活动
  • Background, 后台
  • Suspend, 挂起

对这5种状态,这里先不过多解释,看下图也许就会明白许多。

iOS应用状态及互相切换

iOS应用程序状态及互相切换

从这个示意图,我们可以看到哪些状态间是可以互相转化的。而在这些状态互相转化的同时,AppDelegate中对应的生命周期方法会被调用:

  • - (BOOL)application:(UIApplication *)application willFinishLaunchingWithOptions:(NSDictionary *)launchOptions      进程启动但还没完成初始化,这个方法是iOS6之后才有的
  • - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions     进程启动基本完成
  • - (void)applicationWillResignActive:(UIApplication *)application    应用程序将要入非活动状态执行,在此期间,应用程序不接收消息或事件
  • - (void)applicationDidBecomeActive:(UIApplication *)application    应用程序入活动状态,这个刚好跟上面那个方法相反
  • - (void)applicationDidEnterBackground:(UIApplication *)application     程序被推送到后台,如果要设置后台继续运行,则在这个函数里面设置即可
  • - (void)applicationWillEnterForeground:(UIApplication *)application   程序从后台将要回到前台
  • - (void)applicationWillTerminate:(UIApplication *)application   程序将要退出

2. 应用一般启动过程

介绍了iOS应用的基本状态和生命周期方法之后,再简要对应用启动和前台状态做一个整理说明。

对于一般的iOS应用来说,启动之后是进入前台运行的,大概的流程如下图:

iOS应用启动并进入前台运行

iOS应用启动并进入前台运行

也就是说通常情况下,应用程序启动,图中的三个方法会被依次调用到。

其中前两个方法是我们需要做一些初始化的地方,只在启动时调用,而后面的BecomeActive方法会因程序中断和前后台切换多次调用到。

对于一个简单的应用,初始化主要做的事情就是把View和Controller初始化并结合起来。此外,必要的情况下还要对传入的Options参数进行解析处理,构建必要的数据结构。而按照官方文档的要求,整个启动过程需要在5秒钟内完成,否则应用进程会因无响应的原因被kill掉。因此,如果有其它任务需要执行,则应该开启主线程之外的线程来进行,或者是推迟到之后更合适的实际来进行。

应用程序也有启动后直接进入后台运作的,这个可以参看苹果官方文档说明。

3. 前台运行的Active与Inactive

在介绍iOS应用状态5种最基本的状态时,我们发现前台运行有两种状态,分别是Inactive和Active状态。大多数情况下,Inactive状态只是其它状态之间切换时短暂的停留状态,如前后台应用切换时,Inactive状态会在Active和Background之间短暂出现。

但也有一些其它情况,Active和Inactive可以在前台运行时互相切换,比如当一个应用安装运行后第一次尝试使用GPS定位,需要获取用户的允许,给出系统的Alert提示,这时应用会从Active切换到Inactive,直到用户确认后再返回Active。再如,用户在应用运行时从状态条向下拉出通知页,也会发生Active和Inactive状态的切换。此外,还有来电/拒绝接听,以及App Switcher/回到原应用的操作等,都不进入Backgroud状态,而只在Active与Inactive之间切换。

Active、Inactive状态和中断响应

Active、Inactive状态和中断响应

4. 其它

除了上面提到的,UIApplicationDelegate也还有很多其它的回调方法。比如隐私数据保护相关的方法,根据一定的数据保护策略,在锁屏和解锁的时候,数据会变得不可用和可用:

  • - (void)applicationProtectedDataWillBecomeUnavailable:(UIApplication *)application
  • - (void)applicationProtectedDataDidBecomeAvailable:(UIApplication *)application

这是在iOS4开始就支持了的。

在iOS6中,还有了支持应用状态Preserve和Restore的API,使得app可以跨launch保留一些状态信息,这些大多都与UI以及Storyboard相关,细节可参考苹果文档。

此外,在通常的开发过程中,还有几点要注意:

  • 设备锁定时,系统认为当前App进入了Background状态
  • 通常情况下,App在Launch过程完成后是直接进入Active的,WillEnterForeground不会被调用
  • 特殊事件处理的情况下启动,可能不进入前台,此时会调用DidEnterBackground
  • openURL有单独的一套流程,请参见小站另一篇文章

有关前后台状态切换、iOS应用在后台运行、多任务并发和事件循环的内容留在后续文正整理总结。

参考路径:http://www.molotang.com/articles/1254.html

阅读更多
个人分类: 开发笔记
上一篇http多请求流程等了解(待整理)
下一篇iOS中消息的传递机制
想对作者说点什么? 我来说一句

没有更多推荐了,返回首页

关闭
关闭
关闭