【改变了后台任务的运行方式】
在iOS6和之前的系统中,系统在用户退出应用后,如果应用正在执行后台任务的话,系统会保持活跃状态直到后台任务完成或者是超时以后,才会进入真正的低功耗休眠状态。
而在iOS7中,后台任务的处理方式发生了改变。系统将在用户锁屏后尽快让设备进入休眠状态,以节省电力,这时后台任务是被暂停的。之后在设备在特定时间进行系统应用的操作被唤醒(比如检查邮件或者接到来电等)时,之前暂停的后台任务将一起进行。就是说,系统不会专门为第三方的应用保持设备处于活动状态。如下图示:
这个变化在不减少应用的后台任务时间长度的情况下,给设备带来了更多的休眠时间,从而延长了续航。对于开发者来说,这个改变更多的是系统层级的变化,对于非网络传输的任务来说,保持原来的用法即可,新系统将会按照新的唤醒方式进行处理;而对于原来在后台做网络传输的应用来说,苹果建议在iOS7中使用NSURLSession
,创建后台的session并进行网络传输,这样可以很容易地利用更好的后台传输API,而不必受限于原来的时长,关于这个具体的我们一会儿再说。
【后台获取(Background
Fetch)】
现在的应用无法在后台获取信息,比如社交类应用,用户一定需要在打开应用之后才能进行网络连接,获取新的消息条目,然后才能将新内容呈现给用户。说实话这个体验并不是很好,用户在打开应用后必定会有一段时间的等待,每次皆是如此。
iOS7中新加入的后台获取就是用来解决这个不足的:后台获取干的事情就是在用户打开应用之前就使app有机会执行代码来获取数据,刷新UI。这样在用户打开应用的时候,最新的内容将已然呈现在用户眼前,而省去了所有的加载过程。
1、勾选Background Fetch。
2、设定获取间隔。如果不对最小后台获取间隔进行设定的话,系统将使用默认值UIApplicationBackgroundFetchIntervalNever
,也就是永远不进行后台获取。当然,-setMinimumBackgroundFetchInterval:
方法接受的是NSTimeInterval,因此你也可以手动指定一个以秒为单位的最小获取间隔。
[[UIApplication sharedApplication]
setMinimumBackgroundFetchInterval:UIApplicationBackgroundFetchIntervalMinimum];
3、实现后台获取代码并通知系统。在AppDelegate里实现-application:performFetchWithCompletionHandler:。
系统将会在执行fetch的时候调用这个方法,然后开发者需要做的是在这个方法里完成获取的工作,然后刷新UI,并通知系统获取结束,以便系统尽快回到休眠状态。应用在前台能完成的工作在这里都能做,唯一的限制是系统不会给你很长时间来做fetch,一般会小于一分钟,而且fetch在绝大多数情况下将和别的应用共用网络连接。这些时间对于fetch一些简单数据来说是足够的了,比如微博的新条目。
4、通知系统获取完成。
方法是调用-application:performFetchWithCompletionHandler:
的handler。这个CompletionHandler接收一个UIBackgroundFetchResult
作为参数,可供选择的结果有UIBackgroundFetchResultNewData
,UIBackgroundFetchResultNoData
,UIBackgroundFetchResultFailed
三种,分别表示获取到了新数据(此时系统将对现在的UI状态截图并更新App
Switcher中你的应用的截屏),没有新数据,以及获取失败。
1 //File: YourAppDelegate.m
2 -(void)application:(UIApplication *)application performFetchWithCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler
3 {
4 UINavigationController *navigationController = (UINavigationController*)self.window.rootViewController;
5
6 id fetchViewController = navigationController.topViewController;
7 if ([fetchViewController respondsToSelector:@selector(fetchDataResult:)]) {
8 [fetchViewController fetchDataResult:^(NSError *error, NSArray *results){
9 if (!error) {
10 if (results.count != 0) {
11 //Update UI with results.
12 //Tell system all done.
13 completionHandler(UIBackgroundFetchResultNewData);
14 } else {
15 completionHandler(UIBackgroundFetchResultNoData);
16 }
17 } else {
18 completionHandler(UIBackgroundFetchResultFailed);
19 }
20 }];
21 } else {
22 completionHandler(UIBackgroundFetchResultFailed);
23 }
24 }
5、后台获取调试
1) 新建Background Fetch Scheme。
2)当应用在后台时,模拟一次后台获取。这个比较简单,在app调试运行时,点击Xcode5的Debug菜单中的Simulate
Background Fetch,即可模拟完成一次获取调用。
参考:http://onevcat.com/2013/08/ios7-background-multitask/