iOS 关于前后台定位的设置

2020-12-04  本文已影响0人  浅浅_e90e

这两天把定位的功能仔细捋了一下,做个总结:

用系统自带的定位功能举例吧

如果应用只有简单的定位需求,一般用系统自带的定位功能就可以解决问题了,使用方法如下:

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];这个方法,这样用户在弹窗里就可以选择授权的权限。

如有不对的地方,烦请多多指正,谢谢~

上一篇下一篇

猜你喜欢

热点阅读