奇技指南
前段时间,笔者和GY哥一起吃饭聊天的时候,GY哥问了笔者一个问题,iOS App 可以后台保活吗?是如何做到后台保活的?当时笔者只想到了可以在后台播放静音的音乐,对于唤醒App,可以考虑使用推送的方式。GY哥提到播放静音文件会影响上线吗?这我就不大知道了……由于没有相关实践,笔者后来在网上查了相关资料,总结了本文。
本文转载自公众号QiShare
笔者查询了相关资料后发现,iOS App可以实现后台保活。
短时间保活的方式有beginBackgroundTaskWithName;
App长时间保活的方式有:播放无声音乐、后台持续定位、后台下载资源、BGTaskScheduler等;
唤醒App的方式有:推送、VoIP等;
导读
本文分为如下几部分:
App 运行状态、及状态变化
App 后台保活方式简介
短时间App后台保活
Background Modes AVAudio,AirPlay,and Picture in Picture
Background Modes Location updates
BGTaskScheduler (iOS13.0+)
1 App 运行状态、及状态变化
不低于iOS13.0的设备端App 运行状态
不低于iOS13.0设备端App 运行状态iOS13.0+的设备,支持多场景,共有上图中的Unattached、Foreground Inactive、Foreground Active、Forground Inactive、Background、Suspended 6种状态。
低于iOS13.0的设备端App 运行状态
低于iOS13.0设备端App 运行状态
上图是低于iOS13.0的设备端App的运行状态,分别是Not Running、Foreground Inactive、Foreground Active、Forground Inactive、Background、Suspended 6种状态。Not Running:指用户没有启动App,或用户Terminate App 后,App处于的状态;其他的五种状态和不低于iOS13.0的设备端App的运行状态意义相同。
App 进入后台状态变化
笔者写了个定时器,定时输出“普通定时器进行中”,可以看到,应用进入后台后,基本上立刻,就没有内容输出了。笔者认为可以认为此时App 已经进入Suspended的状态。
App 进入后台下边笔者介绍下,尝试的App后台保活方式。
2
iOS App 后台保活方式简介
短时间App后台保活
beginBackgroundTaskWithName 和 endBackgroundTask笔者尝试过使用相关API,测试过2款手机。对于系统版本低于iOS13(iOS 12.3)的设备(iPhone6 Plus)后台运行时间约3分钟(175秒);
对于系统版本不低于iOS13(
iOS 13.0)的设备(iPhone6 Plus)后台运行时间约31秒;
播放无声音乐
App 进入后台后,播放无声音乐,适用于音视频类App。笔者对逆向不了解,从iOS项目技术还债之路《一》后台下载趟坑中得知,腾讯视频、爱奇艺采用了播放无声音乐保活的方式。
后台持续定位
对于定位类App,持续定位App,可以实现App后台保活。定位类App需要后台保活,像系统的地图应用,在导航的时候切换App的时候,就需要后台保活。
后台下载资源
对于需要下载资源的App,需要后台下载资源,如我们在某App下载资源的时候,我们希望在切换App时候,或者App退出后台后,资源仍然继续下载,这样当我们打开App的时候,资源已经下载好了。
BackgroundTasks
BackgroundTasks.framework 是iOS13新增的framework,笔者认为此framework中的API可以在信息流类的App中发挥作用。
3
短时间App后台保活
系统版本低于iOS13.0的设备
系统版本低于iOS13.0的设备,在应用进入后台的时候,开始后台任务([[UIApplication sharedApplication] beginBackgroundTaskWithName:)。在应用进入前台时或后台任务快过期的回调中,终止后台任务([[UIApplication sharedApplication] endBackgroundTask:)。
示例代码如下:
- (void)applicationDidEnterBackground:(UIApplication *)application {
- (void)applicationWillEnterForeground:(UIApplication *)application {
添加相关代码后,笔者在iOS12.4的6 Plus上测试结果如下,应用在进入后台后,大概还运行了175秒。
2019-12-29 19:06:55.647288+0800 QiAppRunInBackground[1481:409744] -[AppDelegate applicationDidEnterBackground:]:应用进入后台DidEnterBackground
系统版本不低于iOS13.0的设备
- (void)sceneDidEnterBackground:(UIScene *)scene API_AVAILABLE(ios(13.0)){
- (void)sceneWillEnterForeground:(UIScene *)scene API_AVAILABLE(ios(13.0)){
添加相关代码后,笔者在iOS13.0的6s上测试结果如下,应用在进入后台后,大概还运行了31秒。
iOS13.0+ App 进入后台
Xs·H 提到过,如果持续后台播放无声音频或是使用后台持续定位的方式实现iOS App后台保活,会浪费电量,浪费CPU,所以一般情况下,使用这种短时间延长App 后台保活的方式,应该够开发者做需要的操作了。
4
Background Modes AVAudio,AirPlay,and Picture in Picture
对于音视频类App,如果需要后台保活App,在App 进入后台后,可以考虑先使用短时间保活App的方式,如果后台保活App方式快结束后,还没处理事情,那么可以考虑使用后台播放无声音乐。相关示例代码如下。
- (AVAudioPlayer *)player {
[self.player prepareToPlay];
系统版本低于iOS13.0的设备
- (void)applicationDidEnterBackground:(UIApplication *)application {
- (void)applicationWillEnterForeground:(UIApplication *)application {
系统版本不低于iOS13.0的设备
- (void)sceneDidEnterBackground:(UIScene *)scene API_AVAILABLE(ios(13.0)){
- (void)sceneWillEnterForeground:(UIScene *)scene API_AVAILABLE(ios(13.0)){
5
Background Modes Location updates 开启后台定位持续更新配置,添加了位置隐私申请后,在应用使用持续定位的情况下,可以实现后台保活App。
添加后台获取位置及音频使用能力
添加获取位置隐私申请对于定位类App,如果需要后台保活App,在用户使用了定位功能后,App 进入后台后,App自动具备后台保活能力,部分示例代码如下。
self.locationManager = [CLLocationManager new];
如果遇到如下异常信息:
2019-12-29 19:57:46.481218+0800 QiAppRunInBackground[1218:141397] 异常:Invalid parameter not satisfying: !stayUp || CLClientIsBackgroundable(internal->fClient) || _CFMZEnabled()
检查:
Signing&Capablities 的 backgounrd Modes 中 Location updates是否勾选;
后台下载资源
当需要实现下载资源类的App在进入后台后,持续下载资源的需求时。我们可能需要使用后台如下示例示例代码。创建指定标识的后台NSURLSessionConfiguration,配置好
NSURL *url = [NSURL URLWithString:@"https://images.pexels.com/photos/3225517/pexels-photo-3225517.jpeg"];
6
BGTaskScheduler(iOS13.0+) 如果我们的App是信息流类App,那么我们可能会使用到BGTaskScheduler.framework中的API,实现后台保活App,帮助用户较早地获取到较新信息。
笔者尝试使用BGTaskScheduler 做了一个获取到App调度的时候。更新首页按钮颜色为随机色并且记录调度时间的Demo。
项目配置
为了App 支持 BGTaskScheduler,需要在项目中配置Background fetch,及Background Processing;
需要在Info.plist文件中添加 key 为Permitted background task scheduler identifiers,Value为数组的内容。Value的数组填写,刷新的任务标识和清理的任务标识。
注册后台任务
在应用启动后,注册后台任务。
- (void)registerBgTask {
调度App 刷新
应用进入后台后,调度App 刷新。
- (void)sceneDidEnterBackground:(UIScene *)scene API_AVAILABLE(ios(13.0)){
得到后台任务调度的时候,调用App刷新的方法,笔者在这个方法中做了发送更新首页按钮颜色的通知,并且记录了当前更新时间的记录。
- (void)handleAppRefresh:(BGAppRefreshTask *)appRefreshTask API_AVAILABLE(ios(13.0)){
经过测试,发现App 在退到后台,没有手动Terminate App的情况下。苹果有调用过App调度任务的方法。现象上来看就是隔一段时间,我们再打开App 的时候可以发现,首页的按钮颜色改变了,相应的日志中追加了,调起相关方法的时间记录。
手动触发后台任务调度
Xcode运行我们的App-> App 退到后台-> 打开App 进入前台-> 点击下图中蓝框中的Pause program execution,输入如下内容
后台模拟调起App
e -l objc -- (void)[[BGTaskScheduler sharedScheduler] _simulateLaunchForTaskWithIdentifier: @"com.qishare.ios.wyw.background.refresh"]
-> 再次点击Continue program execution,就可以模拟后台启动任务,调用我们的App。
Continue program execution
查看日志记录小提示
之前记得听沐灵洛提过怎么便于查看日志,正好我这里也用到了。便于我们可以直接在File App中查看写入到我们App的Documents中的文件,可以在Info.plist文件中添加key为LSSupportsOpeningDocumentsInPlace ,value为YES的键值对App 接入 iOS 11 的 Files App。
经过我们操作后,就可以打开File App -> 浏览 -> 我的iPhone -> 查看选择我们的App -> 查看我们的日志记录文件。
7
示例Demo
https://github.com/QiShare/QiAppRunInBackground