写在前面:直接上代码。
//
// ViewController.h
// 定位权限
//
// Created by zengchunjun on 16/10/11.
// Copyright © 2016年 zengchunjun. All rights reserved.
//
#import <UIKit/UIKit.h>
@interface ViewController : UIViewController
@end
//
// ViewController.m
// 定位权限
//
// Created by zengchunjun on 16/10/11.
// Copyright © 2016年 zengchunjun. All rights reserved.
//
#import "ViewController.h"
#import <CoreLocation/CoreLocation.h>
@interface ViewController ()<CLLocationManagerDelegate>
@property (nonatomic,strong)CLLocationManager *locationManager;
@end
@implementation ViewController
// 懒加载
- (CLLocationManager *)locationManager
{
if (!_locationManager) {
_locationManager = [[CLLocationManager alloc] init];
_locationManager.delegate = self;
}
return _locationManager;
}
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
}
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
// 请求授权
/**
* 使用时授权方式,就是APP运行在前台时可以使用获取地理等服务,一旦运行在后台,APP将不能使用该服务,如果想使用,可以再info里面的capabilities下的background Modes 下的location updates勾上,iOS 9以后还需加上一句,但是这样又有一个问题,就是退到后台后,屏幕上方有一个在使用定位服务的提示。
*/
// 前台授权方式,需要在info.plist里面添加 NSLocationWhenInUseUsageDescription 描述
[self.locationManager requestWhenInUseAuthorization];
if ([[[UIDevice currentDevice] systemVersion] floatValue] > 9.0)
self.locationManager.allowsBackgroundLocationUpdates = YES;
// 前台后台授权方式 需要在info.plist里面添加NSLocationAlwaysUsageDescription 描述
// [self.locationManager requestAlwaysAuthorization];
[self.locationManager startUpdatingLocation];
}
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations
{
NSLog(@"%s",__func__);
}
- (void)locationManager:(CLLocationManager *)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status
{
/**
* kCLAuthorizationStatusNotDetermined = 0,
kCLAuthorizationStatusRestricted,
kCLAuthorizationStatusDenied,
kCLAuthorizationStatusAuthorizedAlways NS_ENUM_AVAILABLE(NA, 8_0),
kCLAuthorizationStatusAuthorizedWhenInUse
*/
switch (status) {
case kCLAuthorizationStatusNotDetermined:
// 弹出授权窗口还未点击允许和不允许时会调用一次
NSLog(@"没有决定");
break;
case kCLAuthorizationStatusRestricted:
NSLog(@"受限制的");
break;
case kCLAuthorizationStatusDenied:
// 手机内设置关闭定位服务或APP内设置不允许使用位置信息
NSLog(@"拒绝的");
// 所以这里还需要判断是关闭的定位服务还是不被允许使用位置信息
if ([CLLocationManager locationServicesEnabled]) {// 打开了定位服务,说明上次使用是不允许的
// 如果是关闭了定位服务,苹果会自定弹出一个窗口,点击设置会跳到相应的地方进行设置
}else{
// 在这里可以设置一个提示框,提示用户拒绝该APP使用位置的服务,如若用户答应使用可以执行以下代码进行跳转到设置界面
NSURL *serv = [NSURL URLWithString:UIApplicationOpenSettingsURLString];
[[UIApplication sharedApplication] openURL:serv];
}
break;
case kCLAuthorizationStatusAuthorizedAlways:
NSLog(@"总是授权,前台后台都可以定位");
break;
case kCLAuthorizationStatusAuthorizedWhenInUse:
NSLog(@"使用时授权,就是在前台使用时可以定位");
break;
default:
break;
}
}
@end
先说懒加载。以上是正确的代码,正确的截图就不上传了,修改一下加载locationManager的get方法,其他代码不变,如下:
// 懒加载
- (CLLocationManager *)locationManager
{
if (!self.locationManager) {
self.locationManager = [[CLLocationManager alloc] init];
self.locationManager.delegate = self;
}
return self.locationManager;
}
运行结果:在get方法处崩溃
这是为什么?一句话解释:自己调用了自己。self.locationManager这个点语法本身就调用了get方法,然而get方法里面又使用了self.locationManager来判断,最后导致崩溃。
接下来说一下使用地理位置服务的授权问题,这个授权细节比较多。
两种授权方式都是iOS 8以后才出的。
前台授权:[self.locationManager requestWhenInUseAuthorization]; info.plist里面添加 NSLocationWhenInUseUsageDescription 描述
前后台授权:[self.locationManager requestAlwaysAuthorization]; info.plist里面添加NSLocationAlwaysUsageDescription 描述
根据实际情况选用一个,具体细节看代码里的注释