在iOS开发中,当程序进入后台或者锁屏时,若程序是用socket进行连接,就有可能与服务器断开连接。这是因为iOS设备为了让设备尽量省电,减少不必要的用电时,对app后台采用了墓碑机制的后台。
墓碑机制简单的理解,就是当任务被中断(进入后台)时,系统记录下当前程序的当前状态后,再中止程序,当程序再次启用的时候,恢复到原来的状态。
这就导致了才要socket进行连接的程序在进入后台时会断开连接,进入前台需要重新连接。所以,根据需要,开发时需要保持后台是“真后台”。也就是程序在后台依然保持运行状态。
实现“真后台”通常使用以下两种方法:
(1)在后台一直放歌(歌要为无声)
(2)在后台进行定位服务
比较两种方法,个人认为,虽然两种方法都是投机的方法,但是第二种比第一种更加合理。因为若程序一直在运行,若用户在听歌、接听电话或者其他需要用到喇叭的操作时可能会收到影响。而定位可以一直不显示,也不影响其他操作。
在此,着重介绍第二种方法在OC中的方法。
因为进入后台等操作都属于APP的生命周期,所以需要在AppDelegate中完成操作。
首先,在AppDelegate的.m文件中导入<CoreLocation/CoreLocation.h>,并加入CLLocationManagerDelegate的代理,并设置所需的东西,示例如下:
#import "AppDelegate.h"
#import <CoreLocation/CoreLocation.h>
@interface AppDelegate ()<CLLocationManagerDelegate>
//定位
@property (strong,nonatomic)CLLocationManager *locationManager;
//后台任务标识符
@property (assign,nonatomic)UIBackgroundTaskIdentifier bgTask;
//终结处理程序的Block
@property (strong,nonatomic)dispatch_block_t expirationHandler;
//是否到达工作期限
@property (assign,nonatomic)BOOL jobExpired;
//是否在后台
@property (assign,nonatomic)BOOL background;
@end
在didFinishLaunchingWithOptions方法中初始化locationManager并设置自动唤醒程序和绑定协议
UIApplication *app = [UIApplication sharedApplication];
__weak AppDelegate *selfRef = self;
self.expirationHandler = ^{ //创建后台自唤醒
[app endBackgroundTask:selfRef.bgTask];
// 后台任务无效
selfRef.bgTask = UIBackgroundTaskInvalid;
// 进入后台任务终结处理
selfRef.bgTask = [app beginBackgroundTaskWithExpirationHandler:selfRef.expirationHandler];
selfRef.jobExpired = YES;
while(selfRef.jobExpired)
{
// 线程睡眠1s
[NSThread sleepForTimeInterval:1];
}
// 开始后台任务类型
[selfRef startBackgroundTask];
};
// 检测后台的状态
[self monitorBatteryStateInBackground];
_locationManager = [[CLLocationManager alloc] init];
_locationManager.delegate = self;
上述中的相关方法
#pragma mark 检测后台的状态
- (void)monitorBatteryStateInBackground
{
self.background = YES;
[self startBackgroundTask];
}
#pragma mark 开始后台任务
- (void)startBackgroundTask
{
if([Api sharedApi].loginedUserId)//当登陆状态才进入后台循环(此处需要自己设定,判断用户是否登录)
{
// 开启异步线程
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
NSInteger count=0;
BOOL NoticeNoBackground=false;//只通知一次标志位
BOOL FlushBackgroundTime=false;//只通知一次标志位
self.locationManager.distanceFilter = kCLDistanceFilterNone;//任何运动均接受,任何运动将会触发定位更新
self.locationManager.desiredAccuracy = kCLLocationAccuracyHundredMeters;//定位精度
while(self.background && !self.jobExpired){
[NSThread sleepForTimeInterval:1];
count++;
if(count>60)//每60s进行一次开启定位,刷新后台时间
{
count=0;
// 开始定位服务
[self.locationManager startUpdatingLocation];
[NSThread sleepForTimeInterval:1];
// 停止定位服务
[self.locationManager stopUpdatingLocation];
FlushBackgroundTime=false;
}
if(![Api sharedApi].loginedUserId)//未登录或者掉线状态下关闭后台(此处需要自己设定,判断用户是否登录)
{
[[UIApplication sharedApplication] endBackgroundTask:self.bgTask];
return;//退出循环
}
NSTimeInterval backgroundTimeRemaining = [[UIApplication sharedApplication] backgroundTimeRemaining];
if(backgroundTimeRemaining<30&&NoticeNoBackground==false)
{
NoticeNoBackground=true;
}
//测试后台时间刷新
if(backgroundTimeRemaining>200&&FlushBackgroundTime==false)
{
[[NSNotificationCenter defaultCenter]postNotificationName:@"MessageUpdate" object:@"刷新后台时间成功\n"];
FlushBackgroundTime=true;
}
}
self.jobExpired = NO;
});
}
}
在生命周期的applicationDidBecomeActive方法中写入
[UIApplication sharedApplication].applicationIconBadgeNumber=0;
// 停止定位服务
[self.locationManager stopUpdatingLocation];
self.background = NO;
通过定位协议的didFailWithError方法来判断定位服务出错的原因,并通过didUpdateLocations方法来获取所定位的位置(注: didUpdateLocations中一定要获取位置的改变,否则后台刷新无用),代码如下:
- (void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error//当定位服务不可用出错时,系统会自动调用该函数
{
// 自由发挥
}
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations
{
CLLocation *loc = [locations lastObject];
float latitudeMe = loc.coordinate.latitude;
float longitudeMe = loc.coordinate.longitude;
NSLog(@"%f,%f",latitudeMe,longitudeMe);
}
最后,希望这篇文章对各位有所帮助。
此文参考《关于iOS后台长时间挂起的方法》