iOS定时器在后台保活及扩展后台永久存活
2019-03-26 本文已影响0人
Mr姜饼
实战出真知 ,下面带大家慢慢玩转后台存活。
首先新建个项目 ,写上几句简单的代码
#import "ViewController.h"
@interface ViewController ()
@property (nonatomic , strong) dispatch_source_t timer;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
__block int flag = 0;
_timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, dispatch_get_main_queue());
dispatch_source_set_timer(_timer, DISPATCH_TIME_NOW, 1 * NSEC_PER_SEC, 1 * NSEC_PER_SEC );
dispatch_source_set_event_handler(_timer, ^{
flag ++;
NSLog(@"flag = %d",flag);
});
dispatch_resume(_timer);
}
接下来 我们运行项目,控制台能循环打印信息 并且累加;这时候我们按下Home键,切到后台,这时候控制台的信息不打印了,这是为什么呢,timer失去作用了。
分析问题
一般来说,没有进行过任何设置的app,默认退到后台极短的几秒后就变成挂起状态
初步解决
利用 UIBackgroundTaskIdentifier 后台任务标记时,让我们来看看后台程序能撑多久
- (void)applicationDidEnterBackground:(UIApplication *)application {
[self startBackgroundTask];
}
- (void)startBackgroundTask{
UIApplication* app = [UIApplication sharedApplication];
__block UIBackgroundTaskIdentifier bgTask ;
bgTask = [app beginBackgroundTaskWithExpirationHandler:^{
//这里延迟的系统时间结束
NSLog(@"%f",app.backgroundTimeRemaining);
}];
}
在appdelegate中加上上述代码 ,运行程序,再切换至后台,接着我们再观察控制台的打印信息,看看什么时候停止打印。
image.png
我们可以看到 后台在三分钟左右的样子 后台任务结束,并且预留了4秒左右的缓冲时间。这时候app休眠了。
方法二次改进
我们利用定位系统保持后台永活
首先我们打开应用位置更新功能
image.png
然后再plist文件中写上
<key>NSLocationAlwaysUsageDescription</key>
<string>Location is required to find out where you are</string>
继续加上如下代码
- (void)applicationDidEnterBackground:(UIApplication *)application {
[self startBackgroundTask];
[self startLocationUpdate];
}
- (void)startLocationUpdate{
if(!_locationManager){
_locationManager = [[CLLocationManager alloc] init];
_locationManager.allowsBackgroundLocationUpdates = YES;
_locationManager.delegate = self;
[_locationManager requestAlwaysAuthorization];
[_locationManager startUpdatingLocation];
}
}
接下来运行再切换后台
等待三分钟看看结果
这时候发现后台控制器一直处于打印状态 ,大功告成。
额外扩展
我们也可以用后台播放视频的方法来做到后台永活的功能,做法与后台定位的方式类似,这里就不多说了
- (void)applicationDidEnterBackground:(UIApplication *)application {
[self stratBadgeNumberCount];
[self startBgTask];
/** 播放声音 */
[self.player play];
}
- (AVAudioPlayer *)player{
if (!_player){
NSURL *url=[[NSBundle mainBundle]URLForResource:@"work5.mp3" withExtension:nil];
_player = [[AVAudioPlayer alloc]initWithContentsOfURL:url error:nil];
[_player prepareToPlay];
//一直循环播放
_player.numberOfLoops = -1;
AVAudioSession *session = [AVAudioSession sharedInstance];
[session setCategory:AVAudioSessionCategoryPlayback error:nil];
[session setActive:YES error:nil];
}
return _player;
}