IOS的生命周期及应用的前后台切换

一、综述

1.app的几个状态
  * 停止运行(Not running):应用程序已经终止或者还没有启动。
  * 不活动(Inactive):应用程序处在前台,但是不再接收事件(eg:用户锁住了设备)。
  * 活动(Active):app处在使用之中。
  * 后台(Background):app不在屏幕上显示,但是它仍然在执行代码。大多数程序进入这个状态后会在这个状态
    上停留一会。时间到了之后会进入挂起状态。有的程序经过特殊的请求可以长期处于Background状态。
  * 挂起(Suspended) :app仍然驻留在内存,但它不执行代码。系统会自动把程序变成这个状态而且不会发出通
    知。当挂起时,程序还是停留在内存,当内存低时,系统就把挂起的程序清除掉,为前台程序提供更多的内存。

2.活动-->后台-->挂起
  1)大部分app通常在几秒内就从后台变成了挂起。
  2)挂起的app仍然驻留在设备的内存内,所以当你返回app时,能快速恢复他们,它们并没有占用cpu时间,
    也不消耗电池电量。
    这些驻留在内存中的app,系统会自动删除它们,当你打开一个消耗大内存的app时,ios会自动清理挂起的app。

3.运行状态代理
// 告诉代理进程启动,但是还没进入状态保存
- (BOOL)application:(UIApplication *)application willFinishLaunchingWithOptions:(NSDictionary *)launchOptions NS_AVAILABLE_IOS(6_0);

// 告诉代理启动基本完成,程序准备开始运行
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions NS_AVAILABLE_IOS(3_0);

// 应用进入非活动状态,在此期间,应用程序不接受消息或时间,比如电话
- (void)applicationWillResignActive:(UIApplication *)application;

// 应用进入活动状态
- (void)applicationDidBecomeActive:(UIApplication *)application;

// 应用被推送到后台,所以要设置后台运行的时候,在这个函数中设置即可
- (void)applicationDidEnterBackground:(UIApplication *)application;

// 应用从后台重新回到前台
- (void)applicationWillEnterForeground:(UIApplication *)application NS_AVAILABLE_IOS(4_0);

// 应用退出时被调用,通常是用来保存数据和一些退出前的清理工作。这个需要设置UIApplicationExitsOnSuspend的键值
- (void)applicationWillTerminate:(UIApplication *)application;

// 当程序载入后执行
- (void)applicatonDidFinishLaunching:(UIApplication *)application;

1)启动程序
    willFinishLaunchingWithOptions
    didFinishLaunchingWithOptions
    applicationDidBecomeActive
2) 点击home键
    applicationWillResignActive
    applicationDidEnterBackground
3) 双击home键,再打开程序
    applicationWillEnterForeground
    applicationDidBecomeActive

二、图解
2.1 加载应用程序进入前台
IOS的生命周期及应用的前后台切换_第1张图片

2.2 加载应用程序进入后台
IOS的生命周期及应用的前后台切换_第2张图片

2.3 对main函数解释

#import <UIKit/UIKit.h>
#import "AppDelegate.h"
int main(int argc, char * argv[]) {
    @autoreleasepool {
        return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
    }
}

UIApplicationMain函数有四个参数,你不需要改变这些参数值,不过我们也需要理解这些参数和程序是如何开始的

argc 和argv参数包含了系统带过来的启动时间。 第三个参数确定了主要应用程序类的名称,这个参数指定为nil,这样UIKit就会使用默认的程序类UIApplication。第四个参数是程序自定义的代理类名,这个类负责系统和代码之间的交互。它一般在Xcode新建项目时会自动生成。

另外 UIApplicationMain函数加载了程序主界面的文件。虽然这个函数加载了界面文件,但是没有放到应用程序的windows上,你需要在Delegate的 application:willFinishLaunchingWithOptions方法中加载它。

一个应用程序可以有一个主的storyboard文件或者有一个主的nib文件,但不能同时有两个存在。

如果程序在启动时没有自动加载主要的故事版或nib文件,你可以在application:willFinishLaunchingWithOptions方法里准备windows的展示。

三、响应中断
3.1 当一个基于警告式的中断发生时,比如有电话打进来了,这是程序会临时进入inactive状态,这用户可以选择如何处理这个中断,流程如下图:
IOS的生命周期及应用的前后台切换_第3张图片
在iOS5,通知不会把程序变成为激活状态,通知会显示在状态栏上,如果你;拉下状态栏,程序会变成inactive,把状态栏放回去,程序变回active。

按锁屏键也是另外一种程序的中断,当你按下锁屏键,系统屏蔽了所有触摸事件,把app放到了后台,这时app状态是 inactive,并进入后台。

3.2 当有这些中断时,我们的app该怎么办呢?我们应该在applicationWillResignActive:方法中:

停止timer 和其他周期性的任务
停止任何正在运行的请求
暂停视频的播放
如果是游戏那就暂停它
减少OpenGL ES的帧率
挂起任何分发的队列和不重要的操作队列(你可以继续处理网络请求或其他时间敏感的后台任务)。
当程序回到active状态 , applicationDidBecomeActive: 方法应该上面提到的任务重新开始,比如重新开始timer, 继续分发队列,提高OpenGL ES的帧率。不过游戏要回到暂停状态,不能自动开始。

四、转到后台运行

S:只有在IOS4以上系统或者支持多任务的设备才能后台运行。不然会直接结束状态。

4.2 当应用程序进入后台时,我们应该做写什么呢?

保存用户数据或状态信息,所有没写到磁盘的文件或信息,在进入后台时,最后都写到磁盘去,因为程序可能在后台被杀死,
释放尽可能释放的内存
applicationDidEnterBackgound: 方法有大概5秒的时间让你完成这些任务。如果超过时间还有未完成的任务,你的程序就会被终止而且从内存中清除。如果还需要长时间的运行任务,可以调用 beginBackgroundTaskWithExpirationHandler 方法去请求后台运行时间和启动线程来运行长时间运行的任务。

4.3 应用程序在后台时的内存使用

在后台时,每个应用程序都应该释放最大的内存。系统努力的保持更多的应用程序在后台同时 运行。不过当内存不足时,会终止一些挂起的程序来回收内存,那些内存最大的程序首先被终止。

事实上,应用程序应该的对象如果不再使用了,那就应该尽快的去掉强引用,这样编译器可以回收这些内存。如果你想缓存一些对象提升程序的性能,你可以在进入后台时,把这些对象去掉强引用。

下面这样的对象应该尽快的去掉强引用:

图片对象
你可以重新加载的 大的视频或数据文件
任何没用而且可以轻易创建的对象
在后台时,为了减少程序占用的内存,系统会自动在回收一些系统帮助你开辟的内存。比如:

系统回收Core Animation的后备存储。

去掉任何系统引用的缓存图片

去掉系统管理数据缓存强引用

五、返回前台运行

当app处于挂起状态时,它是不能执行任何代码的。因此它不能处理在挂起期间发过来的通知,比如方向改变,时间改变,设置的改变还有其他影响程序展现的或状态的通知。在程序返回后台或前台是,程序都要正确的处理这些通知。

六、程序的终止
程序只要符合以下情况之一,只要进入后台或挂起状态就会终止:

iOS4.0以前的系统

app是基于iOS4.0之前系统开发的。

设备不支持多任务

在Info.plist文件中,程序包含了 UIApplicationExitsOnSuspend 键。

app如果终止了 ,系统会调用app的代理的方法 applicationWillTerminate: 这样可以让你可以做一些清理工作。你可以保存一些数据或app的状态。这个方法也有5秒钟的限制。超时后方法会返回程序从内存中清除。

注意:用户可以手工关闭应用程序。

七、 The Main Run Loop 主运行循环

Main Run Loop负责处理用户相关的事件。UIApplication对象在程序启动时启动main run Loop,它处理事件和更新视图的界面。看Main Run Loop就知道,它是运行在程序的主线程上的。这样保证了接收到用户相关操作的事件是按顺序处理的。

用户操作设备,相关的操作事件被系统生成并通过UIKit的指定端口分发。事件在内部排成队列,一个个的分发到Main run loop 去做处理。UIApplication对象是第一个接收到时间的对象,它决定事件如何被处理。触摸事件分发到主窗口,窗口再分发到对应出发触摸事件的View。其他的事件通过其他途径分发给其他对象变量做处理。

大部分的事件可以在你的应用里分发,类似于触摸事件,远程操控事件(线控耳机等)都是由app的 responder objects 对象处理的。Responder objects 在你的app里到处都是,比如:UIApplication 对象。view对象,view controller 对象,都是resopnder objects。大部分事件的目标都指定了resopnder object,不过事件也可以传递给其他对象。比如,如果view对象不处理事件,可以传给父类view或者view controller。

你可能感兴趣的:(ios,APP,应用)