iOS 关于前后台定位的设置
这两天把定位的功能仔细捋了一下,做个总结:
用系统自带的定位功能举例吧
如果应用只有简单的定位需求,一般用系统自带的定位功能就可以解决问题了,使用方法如下:
1、导入
CoreLocation.framework
2、使用的类里引入头文件
#import <CoreLocation/CoreLocation.h>
3、遵守定位的代理
<CLLocationManagerDelegate>
4、声明定位功能管理对象
@property (nonatomic, strong) CLLocationManager* locationManager;
5、检测定位授权状态
- (void)checkServiceStatus
{
if([CLLocationManager locationServicesEnabled]) {
CLAuthorizationStatus status = [CLLocationManager authorizationStatus];
//确定用户的位置服务启用
if(status == kCLAuthorizationStatusNotDetermined){
//用户未选择,继续请求授权
[self requestLocationServicesAuthorization];
}else if(status == kCLAuthorizationStatusRestricted){
//功能受限制,尝试提示用户进入设置页面看一下(一般不会返回该值)
}else if (status == kCLAuthorizationStatusDenied){
//位置服务是在设置中禁用,用户选择了“不允许”
UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"提示" message:@"位置权限未开启,请到设置-隐私-定位中开启" preferredStyle:UIAlertControllerStyleAlert];
UIAlertAction *cancelAction = [UIAlertAction actionWithTitle:@"暂不" style:UIAlertActionStyleCancel handler:nil];
UIAlertAction *okAction = [UIAlertAction actionWithTitle:@"去设置" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
//进入系统设置页面,APP本身的权限管理页面
[[UIApplication sharedApplication] openURL:[NSURL URLWithString:UIApplicationOpenSettingsURLString] options:@{} completionHandler:nil];
}];
[alert addAction:cancelAction];
[alert addAction:okAction];
[self presentViewController:alert animated:YES completion:nil];
}else if(status == kCLAuthorizationStatusAuthorizedWhenInUse){
//已授权使用
}else if(status == kCLAuthorizationStatusAuthorizedAlways) {
//已授权始终使用
}
}
}
6、授权开启定位功能
//授权开启定位功能
- (void)requestLocationServicesAuthorization {
if (!self.locationManager) {
//初始化管理器
self.locationManager = [[CLLocationManager alloc] init];
self.locationManager.delegate = self;
self.locationManager.desiredAccuracy = kCLLocationAccuracyBest;
self.locationManager.distanceFilter = 100.0f;
}
[self.locationManager requestAlwaysAuthorization];
[self.locationManager requestWhenInUseAuthorization];
//开始定位
[self.locationManager startUpdatingLocation];
}
7、target中的设置
(1)info.plist中添加定位相关的key
如果只需要前台定位功能,无需开启后台定位,则只需要添加一个key即可:NSLocationWhenInUseUsageDescription,对应的value是字符串,就是弹出框的副标题,显示给用户看的,比如“App需要访问您的位置信息,以便为您提供最优服务”之类的,自己定义即可。
如果前后台定位都需要开启,为了考虑iOS系统版本的问题,需要再添加两个key:NSLocationAlwaysAndWhenInUseUsageDescription(适用于iOS11之前的后台定位) 和 NSLocationAlwaysUsageDescription (iOS11以后要使用后台定位必加),三个key对应的value内容最好都保持一致。
(2)如果需要启用后台定位功能,需要操作这一步
image.png
弹框输入background Modes,找到这个选项,双击一下就添加进去了
image.png
这个选项需要勾选。
对了,顺便说一下,以前远程推送的服务也是点击加号后自己去找到push notifications这一项,双击添加的。
8、定位的代理方法
#pragma mark - ************** 定位的代理回调方法 **************************
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations {
// 1.获取用户位置的对象
CLLocation *location = [locations lastObject];
CLLocationCoordinate2D coordinate = location.coordinate;
NSLog(@"纬度:%f 经度:%f", coordinate.latitude, coordinate.longitude);
// 2.停止定位
[manager stopUpdatingLocation];
CLLocation *newLocation = [locations firstObject];
CLGeocoder *geocoder = [[CLGeocoder alloc]init];
[geocoder reverseGeocodeLocation:newLocation completionHandler:^(NSArray<CLPlacemark *> *_Nullable placemarks, NSError * _Nullable error) {
for (CLPlacemark *place in placemarks) {
NSLog(@"name,%@",place.name); // 位置名
NSLog(@"thoroughfare,%@",place.thoroughfare); // 街道
NSLog(@"subThoroughfare,%@",place.subThoroughfare);// 子街道
NSLog(@"locality,%@",place.locality); // 市
NSLog(@"subLocality,%@",place.subLocality); // 区
NSLog(@"country,%@",place.country); // 国家
}
}];
}
//定位失败
- (void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error {
if (error.code == kCLErrorDenied) {
// 提示用户出错原因,kCLErrorDenied是个枚举值,可查看其它错误原因
}
}
至此,定位的简单应用已完成,后面是一些对细节的讲解:
针对第六步中的两个方法:
[self.locationManager requestAlwaysAuthorization];
[self.locationManager requestWhenInUseAuthorization];
(1)如果项目中前后台定位都需要使用,直接调用上面这个方法即可,plist中的key值也需要添加那三个。
(2)如果只需要前台定位,那就调用下面的方法,key值只写一个使用中的就行。
(3)这两个方法的作用就是主动调起系统的定位权限弹窗,一定要在调用startUpdatingLocation
方法前调用这个两个方法其中的一个,如果都不调,则无法弹出权限弹窗。
(4)如果key里面配置了后台定位功能,但你调用的是下面[self.locationManager requestWhenInUseAuthorization];
方法,则弹窗提示询问用户是否在应用使用期间定位,选项也只有允许和不允许两个,用户如果要开启后台定位,只能自己去设置-隐私-定位中找到app,把定位权限从使用期间切换到始终允许上面,弹窗里不能选择始终允许,所以如果需要使用后台定位,务必定位前要调用上面[self.locationManager requestAlwaysAuthorization];
这个方法,这样用户在弹窗里就可以选择授权的权限。
如有不对的地方,烦请多多指正,谢谢~