iOS开发-锁屏时断开连接解决方法小析

在iOS开发中,当程序进入后台或者锁屏时,若程序是用socket进行连接,就有可能与服务器断开连接。这是因为iOS设备为了让设备尽量省电,减少不必要的用电时,对app后台采用了墓碑机制的后台。
墓碑机制简单的理解,就是当任务被中断(进入后台)时,系统记录下当前程序的当前状态后,再中止程序,当程序再次启用的时候,恢复到原来的状态。
这就导致了才要socket进行连接的程序在进入后台时会断开连接,进入前台需要重新连接。所以,根据需要,开发时需要保持后台是“真后台”。也就是程序在后台依然保持运行状态。
实现“真后台”通常使用以下两种方法:
(1)在后台一直放歌(歌要为无声)
(2)在后台进行定位服务
比较两种方法,个人认为,虽然两种方法都是投机的方法,但是第二种比第一种更加合理。因为若程序一直在运行,若用户在听歌、接听电话或者其他需要用到喇叭的操作时可能会收到影响。而定位可以一直不显示,也不影响其他操作。
在此,着重介绍第二种方法在OC中的方法。
因为进入后台等操作都属于APP的生命周期,所以需要在AppDelegate中完成操作。
首先,在AppDelegate的.m文件中导入,并加入CLLocationManagerDelegate的代理,并设置所需的东西,示例如下:

#import "AppDelegate.h"
#import 

@interface AppDelegate ()
//定位
@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后台长时间挂起的方法》

你可能感兴趣的:(iOS开发-锁屏时断开连接解决方法小析)