iOS为了让设备尽量省电,减少不必要的开销,保持系统流畅,因而对后台机制采用墓碑式的“假后台”。除了系统官方极少数程序可以真后台,一般开发者开发出来的应用程序后台受到以下限制:
1.用户按Home之后,App转入后台进行运行,此时拥有180s后台时间(iOS7)或者600s(iOS6)运行时间可以处理后台操作
2.当180S或者600S时间过去之后,可以告知系统未完成任务,需要申请继续完成,系统批准申请之后,可以继续运行,但总时间不会超过10分钟。
3.当10分钟时间到之后,无论怎么向系统申请继续后台,系统会强制挂起App,挂起所有后台操作、线程,直到用户再次点击App之后才会继续运行。
当然iOS为了特殊应用也保留了一些可以实现“真后台”的方法,摘取比较常用的:
1.VOIP
2.定位服务
3.后台下载
4.在后台一直播放无声音乐(容易受到电话或者其他程序影响,所以暂未考虑)
5….更多
今天我要讲述的是运用第2种来实现后台长时间运行
运用定位服务进行后台长时间运行方案:
首先要做的就是在info.plist表中添加权限的字段 Privacy - Location When In Use Usage Description 和 Privacy - Location Always Usage Description ,当然还要添加后台运行的字段:
#import
#import
@interface LocationManager : NSObject
@property (nonatomic,strong)CLLocationManager *_locationManager;
+ (LocationManager *)shareInstance;
- (void)startLocation;
#import "LocationManager.h"
@implementation LocationManager
+(LocationManager *)shareInstance
{
static LocationManager *manager = nil;
static dispatch_once_t once;
dispatch_once(&once, ^{
manager = [[LocationManager alloc] init];
manager._locationManager = [[CLLocationManager alloc] init];
//设置iOS设备是否可暂停定位来节省电池的电量。如果该属性设为“YES”,则当iOS设备不再需要定位数据时,iOS设备可以自动暂停定位。 如果不设置为NO,只能在后台运行15分钟后就会处于挂起状态。
manager._locationManager.pausesLocationUpdatesAutomatically = NO;
//这是iOS9中针对后台定位推出的新属性 不设置的话 可是会出现顶部蓝条的哦(类似热点连接)
if ([manager._locationManager respondsToSelector:@selector(allowsBackgroundLocationUpdates)]) {
// 若不设置,默认为NO。不会后台进行定位。
[manager._locationManager setAllowsBackgroundLocationUpdates:YES];
}
});
return manager;
}
- (void)startLocation
{
[self._locationManager startUpdatingLocation];
}
#import "ViewController.h"
#import
#import "LocationManager.h"
@interface ViewController ()
@end
@implementation ViewController
{
LocationManager *locationManager;
NSTimer *timer;
}
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
[self initLocation];
}
- (void)initLocation
{
locationManager = [LocationManager shareInstance];
if (![CLLocationManager locationServicesEnabled]) {
NSLog(@"定位服务当前可能尚未打开,请设置打开!");
return;
}
//如果没有授权则请求用户授权
if ([CLLocationManager authorizationStatus]==kCLAuthorizationStatusNotDetermined){
[locationManager._locationManager requestAlwaysAuthorization];
}else if([CLLocationManager authorizationStatus]==kCLAuthorizationStatusAuthorizedAlways){
//设置代理
locationManager._locationManager.delegate=self;
//设置定位精度
locationManager._locationManager.desiredAccuracy=kCLLocationAccuracyBest;
//定位频率,每隔多少米定位一次
CLLocationDistance distance= 0;//十米定位一次
locationManager._locationManager.distanceFilter=distance;
// 启动跟踪定位
[locationManager._locationManager startUpdatingLocation];
}
}
-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations{
CLLocation *location=[locations firstObject];//取出第一个位置
CLLocationCoordinate2D coordinate=location.coordinate;//位置坐标
NSLog(@"经度:%f,纬度:%f,海拔:%f,航向:%f,行走速度:%f",coordinate.longitude,coordinate.latitude,location.altitude,location.course,location.speed);
if (timer == nil) {
timer = [NSTimer scheduledTimerWithTimeInterval:1.0f target:self selector:@selector(printCurrentTime:) userInfo:nil repeats:YES];
}
}
-(void)printCurrentTime:(id)sender{
NSLog(@"当前的时间是---%@---",[self getCurrentTime]);
}
-(NSString *)getCurrentTime{
NSDateFormatter *dateFormatter=[[NSDateFormatter alloc]init];
[dateFormatter setDateFormat:@"yyyy-MM-DD HH:mm:ss"];
NSString *dateTime=[dateFormatter stringFromDate:[NSDate date]];
return dateTime;
}
ok搞定!
demo地址:后台长时间运行