是一个进入或者离开的动作必威,移动app能解决

CoreLocation框架在LBS类项目中应用广泛,程序员可以用CoreLocation框架轻松的实现地图/定位功能.

CoreLocation框架

简介

CoreLocation框架

一. 简介

在我们日常生活中时常用到地图和定位功能,来导航去你想去的地方或者寻找周边的景点,餐厅,电影院等等,在iOS开发中,要想加入这两大功能,必须基于两个框架进行开发,有了这两个框架,想去哪就去哪。
CoreLocation :用于地理定位,地理编码,区域监听等(着重功能实现)
MapKit :用于地图展示,例如大头针,路线、覆盖层展示等(着重界面展示)

一. iOS8.0之前的定位

一. iOS8.0之前的定位(✨✨✨✨✨)

在移动互联网时代,移动app能解决用户的很多生活琐事,比如

一. iOS8.0之前的定位

二. CoreLocation框架的基本使用

  1. 前台定位
  1. 前台定位
    导入CoreLocation框架以及对应的主头文件

周边:找餐馆、找KTV、找电影院等等

1. 前台定位

  1. 导入CoreLocation框架以及对应的主头文件

     #import <CoreLocation/CoreLocation.h>
    
  2. 创建CLLocationManager对象并设置代理

     self.locationM = [[CLLocationManager alloc] init];
     self.locationM.delegate = self;
    
  3. 调用方法,开始更新用户位置信息

     [self.locationM startUpdatingLocation];
    
  4. 在对应的代理方法中获取位置信息

     -(void)locationManager:(nonnull CLLocationManager *)manager didUpdateLocations:(nonnull NSArray<CLLocation > *)locations  
     {
         NSLog(@"每当请求到位置信息时, 都会调用此方法");  
     }
    

1. CoreLocation使用步骤

  1. 导入CoreLocation框架。
  2. 创建CLLocationManager管理者对象。
  3. 遵循代理,并实现代理方法。
  4. 设置获取用户前后台定位授权
  5. 开始定位。

import <CoreLocation/CoreLocation.h>

创建CLLocationManager对象并设置代理

 self.locationM = [[CLLocationManager alloc] init];
 self.locationM.delegate = self;
调用方法,开始更新用户位置信息

 [self.locationM startUpdatingLocation];
在对应的代理方法中获取位置信息

 -(void)locationManager:(nonnull CLLocationManager *)manager didUpdateLocations:(nonnull NSArray<CLLocation > *)locations  
 {
     NSLog(@"每当请求到位置信息时, 都会调用此方法");  
 }
  1. 后台定位
    在前台定位基础上, 勾选后台模式Location updates image
  2. 额外设置
    每隔多少米定位一次

代码: self.locationM.distanceFilter = 100;
功能: 只有当最新的位置与上一次获取的位置之间的距离, 大于这个值时, 才会通过代理告诉外界
设置定位精确度

代 码: self.locationM.desiredAccuracy = kCLLocationAccuracyBest;
功 能: 通过设置此属性, 获取不同精确度的位置信息
注意事项: 精确度越高,越耗电,定位所需时间越长
枚举注解:
枚举值 含义

kCLLocationAccuracyBestForNavigation    最适合导航
kCLLocationAccuracyBest 精度最好的
kCLLocationAccuracyNearestTenMeters 附近10米
kCLLocationAccuracyHundredMeters    附近100米
kCLLocationAccuracyKilometer    附近1000米
kCLLocationAccuracyThreeKilometers  附近3000米
  1. 知识补充

  2. 定位常识

    1. 标准定位服务(基于gps/基站/wifi定位, 具体使用哪种,苹果有自己规则)

      程序关闭,就没法获取位置
      2) 显著的位置变化定位服务(使用基站进行定位,所以必须要求设备有电话模块)
      > 当app被完全关闭时,也可以接收到位置通知,并让app进入到后台处理
      > 定位精度相比于上面,精度不大,所以耗电小,而且定位更新频率依据基站密度而定

  3. 应用场景

    1. 如果要求定位及时,精度较高,并且运行时间较短,可使用标准定位;
    2. 如果长时间监控用户位置,用户移动速度比较快(例如打车软件),可使用后者
  4. 测试环境:

  5. XCode7.0之前版本,例如XCode6.4版本

  6. 模拟器选择iOS8.0之前的版本

    • 原因 : XCode7.0(包含7.0)之后不支持iOS8.0之前的模拟器
  7. 常见问题总结

  8. 定位不到, 对应的代理方法不执行
    首先,检查运行的模拟器是否是iOS8.0之前的系统版本
    其次,检查模拟器是否设置位置数据
    第三,确保代码无问题(一般都是代理没有设置,或者位置管理器对象是局部变量,亦或是位置管理器对象没有被强引用)
    第四,绝逼是模拟器BUG, 请重置模拟器(是重置,不是重启)
    二. iOS8.0之后定位(✨✨✨✨✨)

  9. 前台定位
    导入CoreLocation框架以及对应的主头文件

 #import <CoreLocation/CoreLocation.h>
创建CLLocationManager对象并设置代理

 self.locationM = [[CLLocationManager alloc] init];
 self.locationM.delegate = self;
请求前台定位授权, 并在Info.Plist文件中配置Key ( Nslocationwheninuseusagedescription )

 [self.locationM requestWhenInUseAuthorization];
调用方法,开始更新用户位置信息

 [self.locationM startUpdatingLocation];
在对应的代理方法中获取位置信息

 -(void)locationManager:(nonnull CLLocationManager *)manager didUpdateLocations:(nonnull NSArray<CLLocation > *)locations  
 {
     NSLog(@"每当请求到位置信息时, 都会调用此方法");  
 }
  1. 后台定位
    方案一:
    在前台定位基础上, 勾选后台模式Location updates image
    注意:此时授权状态如果是前台定位, 那么当APP退到后台时, 屏幕顶部会出现蓝条
    方案二:
    请求前后台定位授权,并在info.plist文件中配置KEY ( NSLocationAlwaysUsageDescription )

[self.locationM requestAlwaysAuthorization];
注意:不需要勾选后台模式, 也可以进行后台定位

注意:此时授权状态如果是前后台定位, 那么即使APP退到后台时, 屏幕顶部会也不会出现蓝条

  1. 监听用户授权状态
    实现以下代理方法即可

// 当用户授权状态发生变化时调用
-(void)locationManager:(nonnull CLLocationManager *)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status
{
switch (status) {
// 用户还未决定
case kCLAuthorizationStatusNotDetermined:
{
NSLog(@"用户还未决定");
break;
}
// 访问受限(苹果预留选项,暂时没用)
case kCLAuthorizationStatusRestricted:
{
NSLog(@"访问受限");
break;
}
// 定位关闭时和对此APP授权为never时调用
case kCLAuthorizationStatusDenied:
{
// 定位是否可用(是否支持定位或者定位是否开启)
if([CLLocationManager locationServicesEnabled])
{
NSLog(@"定位开启,但被拒");
// 在此处, 应该提醒用户给此应用授权, 并跳转到"设置"界面让用户进行授权
// 在iOS8.0之后跳转到"设置"界面代码
NSURL *settingURL = [NSURL URLWithString:UIApplicationOpenSettingsURLString];
if([[UIApplication sharedApplication] canOpenURL:settingURL])
{
[[UIApplication sharedApplication] openURL:settingURL];
}
}else
{
NSLog(@"定位关闭,不可用");
}
break;
}
// 获取前后台定位授权
case kCLAuthorizationStatusAuthorizedAlways:
// case kCLAuthorizationStatusAuthorized: // 失效,不建议使用
{
NSLog(@"获取前后台定位授权");
break;
}
// 获得前台定位授权
case kCLAuthorizationStatusAuthorizedWhenInUse:
{
NSLog(@"获得前台定位授权");
break;
}
default:
break;
}
}

  1. 测试环境:
  2. XCode版本无要求
  3. 模拟器选择iOS8.0之后的版本
  4. 常见问题总结
  5. 定位不到, 对应的代理方法不执行
    首先,检查是否请求授权, 并设置了对应的KEY
    其次,检查模拟器是否设置位置数据
    第三,确保代码无问题(一般都是代理没有设置,或者位置管理器对象是局部变量,亦或是位置管理器对象没有被强引用)
    第四,绝逼是模拟器BUG, 请重置模拟器(是重置,不是重启)
    三. iOS9.0 定位补充(✨✨✨)
  6. 定位变化
  • 前台定位
    (同iOS8.0之后一致, 无任何变化, 都需要主动请求授权)
  • 后台定位
    方案一:
    在前台定位基础上, 勾选后台模式Location updates, 并且设置以下属性为YES

if ([[UIDevice currentDevice].systemVersion floatValue] >= 9.0)
{
self.locationM.allowsBackgroundLocationUpdates = YES;
}
方案二:
请求前后台定位授权,并在info.plist文件中配置KEY ( NSLocationAlwaysUsageDescription )

[self.locationM requestAlwaysAuthorization];

  1. 新的API
    单次定位请求;

代 码: [self.locationM requestLocation];
功 能: 获取一次位置信息
实现逻辑:
(1) 按照定位精确度从低到高进行排序,逐个进行定位.如果在有效时间内, 定位到了精确度最好的位置, 那么就把对应的位置通过代理告知外界.
(2) 如果获取到的位置不是精确度最高的那个,也会在定位超时后,通过代理告诉外界.
注意事项:
(1) 必须实现代理的-locationManager:didFailWithError:方法
(2) 不能与startUpdatingLocation方法同时使用

  1. 测试环境:
  2. XCode版本要求7.0版本以上
  3. 模拟器选择iOS9.0之后的版本
  4. 常见问题总结
  5. 单次定位在模拟器上测试不出效果?
    答: 因为模拟器的位置是固定的, 所以无法测试出效果, 请使用真机进行测试.
    四. CLLocation对象详解(✨✨✨✨✨)
  6. 属性解释

    coordinate : 当前位置所在的经纬度数据
    altitude : 海拔
    speed : 当前速度
    course : 航向(设备移动的方向, 值域范围:0.0 ~ 259.9, 正北方向为0.0)

  7. 重要方法
    代码: - (CLLocationDistance)distanceFromLocation:(CLLocation *)location
    作用: 计算两个位置对象之间的物理距离, 单位是(米)

  8. 场景演练
  9. 场景演示:打印当前用户的行走方向,偏离角度以及对应的行走距离,
    例如:”北偏东30度方向,移动了8米”
  10. 实现步骤:
    1> 获取对应的方向偏向(例如”正东”,”东偏南”)
    2> 获取对应的偏离角度(并判断是否是正方向)
    3> 计算行走距离
    4> 打印信息
  11. 注意事项
    使用位置前, 务必判断当前获取的位置是否有效

代码: if (location.horizontalAccuracy < 0) return;
功能: 如果水平精确度小于零, 代表虽然可以获取位置对象, 但是数据错误, 不可用

  • 经验小结
    一. 定位的应用场景

    1. 导航
    2. 电商APP,获取用户所在城市(需要与(反)地理编码联合使用)
    3. 数据采集用户信息(例如,统计app使用分布)
    4. 查找周边(周边好友, 周边商家等等)
      二. 开发经验
      由于定位非常耗电; 所以为了给用户省电, 你可以遵守以下小经验

    1)不需要获取用户位置时,一定要关闭定位服务:
    2)如果能满足项目需求,尽可能的使用”监听显著位置变化”的定位服务(打车app)
    3)如果可以,尽可能使用低精度的desiredAccuracy
    4)如果是数据采集,(一般都是周期性的去轮询用户位置),在轮询期间一定要关闭定位
    五. 指南针效果实现(✨✨)

  1. 实现思路
    利用"磁力计"传感器,获取设备朝向
    根据设备朝向反向旋转"指南针"图片
  2. 代码实现
    获取设备朝向
  1. 导入CoreLocation框架以及对应的主头文件

    import <CoreLocation/CoreLocation.h>

  2. 创建CLLocationManager对象并设置代理
    self.locationM = [[CLLocationManager alloc] init];
    self.locationM.delegate = self;

  3. 调用方法, 开始获取设备朝向
    [self.locationM startUpdatingHeading];

  4. 在对应的代理方法中获取设备朝向信息
    -(void)locationManager:(CLLocationManager *)manager didUpdateHeading:(CLHeading *)newHeading
    {
    // 旋转图片代码
    }
    旋转图片

// 1.判断当前的角度是否有效(如果此值小于0,代表角度无效)
if(newHeading.headingAccuracy < 0)
return;

// 2.获取当前设备朝向(磁北方向)
CGFloat angle = newHeading.magneticHeading;

// 3.转换成为弧度
CGFloat radian = angle / 180.0 * M_PI;

// 4.带动画反向旋转指南针
[UIView animateWithDuration:0.5 animations:^{
self.compassView.transform = CGAffineTransformMakeRotation(-radian);
}];

  1. 概念补充
    磁北角度: newHeading.magneticHeading ------- 相对于"磁北方向"产生的角度
    真北角度: newHeading.trueHeading ------- 相对于"真北方向"产生的角度
  2. 注意事项

    1. 获取设备朝向前, 先判断"磁力计"是否可用
      [CLLocationManager headingAvailable];

    2. 获取朝向前, 判断当前朝向信息是否有效
      if(newHeading.headingAccuracy < 0) return;

    3. 注意与"航向"的区别
      设备朝向是指手机的朝向; "航向"可以理解为设备的移动方向

    4. 使用"磁力计"传感器获取设备朝向, 不需要请求用户授权
      因为设备朝向不涉及用户隐私
      5.测试环境

    5. XCode版本无要求(建议:XCode7.0不需要开发者账号也可以进行真机调试)

    6. 必须要求真机设备(只有真机设备才有"磁力计"传感器)
      六. 区域监听(✨✨✨)
      1.概念解释
      区 域 : 就是指划定的一块地域范围(比如圆形区域, 则由区域中心, 和半径组成)
      区域监听 : 是指,我们通过代码指定一个区域, 然后当用户持握设备进入或者离开指定区域, 我们都能监听到.

  3. 监听指定区域
    导入CoreLocation框架以及对应的主头文件

 #import <CoreLocation/CoreLocation.h>
创建CLLocationManager对象并设置代理

 self.locationM = [[CLLocationManager alloc] init];
 self.locationM.delegate = self;
请求前后台定位, 或前台定位授权, 并在Info.Plist文件中配置相应的Key

 [self.locationM requestAlwaysAuthorization];
 // [self.locationM requestWhenInUseAuthorization]; 
创建一个区域, 并开始监听

 // 1. 判断区域监听服务是否可用(定位服务是否关闭, 定位是否授权, 是否开启飞行模式)
 if ([CLLocationManager isMonitoringAvailableForClass:[CLCircularRegion class]]) 
 {

     // 创建区域中心
     CLLocationCoordinate2D center = CLLocationCoordinate2DMake(29.12345, 131.23456);

     // 创建区域(指定区域中心,和区域半径)
     CLLocationDistance radius = 1000;

     // 判断区域半径是否大于最大监听区域半径,如果大于, 就没法监听
     if (radius > self.locationM.maximumRegionMonitoringDistance) {
         radius = self.locationM.maximumRegionMonitoringDistance;
     }
     CLCircularRegion *region = [[CLCircularRegion alloc] initWithCenter:center radius:radius identifier:@"小码哥"];

     // 开始监听指定区域
     [self.locationM startMonitoringForRegion:region];
 }
 else
 {
     NSLog(@"区域监听不可用");
 }
在对应的代理方法中监听区域状态

 // 进去监听区域后调用(调用一次)
 -(void)locationManager:(nonnull CLLocationManager *)manager didEnterRegion:(nonnull CLRegion *)region
 {
     NSLog(@"进入区域---%@", region.identifier);
     [manager stopMonitoringForRegion:region];
 }

 // 离开监听区域后调用(调用一次)
 -(void)locationManager:(nonnull CLLocationManager *)manager didExitRegion:(nonnull CLRegion *)region
 {
     NSLog(@"离开区域---%@", region.identifier);
 }
  1. 获取某个区域的当前状态
    监听某个区域时, 只有进入或者离开这个区域时, 才能回调对应的方法, 是一个进入或者离开的动作 如果想知道某一个区域的当前状态(识别用户是在区域内部, 还是区域外部), 则需要使用以下方法
    [self.locationM requestStateForRegion:region];
回调代理: 
    // 请求某个区域状态时, 回调的代理方法
    -(void)locationManager:(CLLocationManager *)manager didDetermineState:(CLRegionState)state forRegion:(CLRegion *)region
{
    switch (state) {
        case CLRegionStateUnknown:
            NSLog(@"未知状态");
            break;
        case CLRegionStateInside:
            NSLog(@"在区域内部");
            break;
        case CLRegionStateOutside:
            NSLog(@"在区域外部");
            break;
        default:
         break;
    }
}
  1. 测试环境
    XCode版本无要求
    iOS模拟器版本无要求

  2. 注意事项

  3. 想要做区域监听, 在iOS8.0之后, 必须请求位置授权
    代码: [self.locationM requestAlwaysAuthorization];
    原因: 区域监听的原理就是获取用户的位置, 然后在判断该位置是否在制定区域内, 所以会涉及到用户隐私(位置), 而在iOS8.0之后, 想要访问用户位置信息, 就需要主动请求授权;

  4. 使用前, 先判断区域监听是否可用
    代码: [CLLocationManager isMonitoringAvailableForClass:[CLCircularRegion class]]

  5. 注意区域半径是否大于最大区域监听半径(如果大于, 则无法监听成功)
    代码: radius > self.locationM.maximumRegionMonitoringDistance

  6. 常见问题

  7. 区域监听, 测试没有效果?
    首先, 确定代码没有问题, 是否有请求授权;
    其次, 尝试修改模拟器位置信息, 触发进入区域或离开区域的动作
    第三, 如果模拟器出现BUG, 定位不到, 也会无法判定当前区域状态; 所以, 最后可以尝试重置模拟器.
    七. (反)地理编码(✨✨✨✨✨)

  8. 概念解释
    地理编码: 是指根据地址关键字, 将其转换成为对应的经纬度等信息;
    发地理编码: 是指根据经纬度信息, 将其转换成为对应的省市区街道等信息;

  9. 地理编码
    导入CoreLocation框架以及对应的主头文件

导航:根据用户设定的起点和终点,进行路线规划,并指引用户如何到达

2. 后台定位

  1. 在前台定位基础上, 勾选后台模式Location updates

Snip20151031_3.png

三. CLLocationManager的使用

学习CLLocationManager可以分为三个部分。1.定位 2.手机朝向 3.区域监听

导入CoreLocation框架以及对应的主头文件 #import <CoreLocation/CoreLocation.h>

创建CLLocationManager对象并设置代理 self.locationM = [[CLLocationManager alloc] init]; self.locationM.delegate = self;

调用方法,开始更新用户位置信息 [self.locationM startUpdatingLocation];在对应的代理方法中获取位置信息 -locationManager:(nonnull CLLocationManager *)manager didUpdateLocations:(nonnull NSArray<CLLocation > *)locations { NSLog(@"每当请求到位置信息时, 都会调用此方法"); }

import <CoreLocation/CoreLocation.h>

创建CLGeocoder对象

self.geoC = [[CLGeocoder alloc] init];
根据地址关键字, 进行地理编码

// 直接根据地址进行地理编码(返回结果可能有多个,因为一个地点有重名)
[self.geoC geocodeAddressString:@"广州" completionHandler:^(NSArray<CLPlacemark *> * __nullable placemarks, NSError * __nullable error)
{
// 包含区,街道等信息的地标对象
CLPlacemark *placemark = [placemarks firstObject];
// 城市名称
NSString *city = placemark.locality;
// 街道名称
NSString *street = placemark.thoroughfare;
// 全称
NSString *name = placemark.name;
}];

  1. 反地理编码
    导入CoreLocation框架以及对应的主头文件

在上述应用中,都用到了定位和地图功能,在iOS开发中,要想加入这2大功能,必须基于2个框架进行开发

3. 额外设置

  1. 每隔多少米定位一次

     代码: self.locationM.distanceFilter = 100;
     功能: 只有当最新的位置与上一次获取的位置之间的距离, 大于这个值时, 才会通过代理告诉外界
    
  2. 设置定位精确度

     代   码: self.locationM.desiredAccuracy = kCLLocationAccuracyBest;
     功   能: 通过设置此属性, 获取不同精确度的位置信息
     注意事项: 精确度越高,越耗电,定位所需时间越长
     枚举注解:   
    
    枚举值 含义
    kCLLocationAccuracyBestForNavigation 最适合导航
    kCLLocationAccuracyBest 精度最好的
    kCLLocationAccuracyNearestTenMeters 附近10米
    kCLLocationAccuracyHundredMeters 附近100米
    kCLLocationAccuracyKilometer 附近1000米
    kCLLocationAccuracyThreeKilometers 附近3000米

1. CLLocationManager -- 定位

  1. 后台定位

import <CoreLocation/CoreLocation.h>

创建CLGeocoder对象

self.geoC = [[CLGeocoder alloc] init];
根据经纬度信息, 进行反地理编码

// 根据经纬度信息进行反地理编码
[self.geoC reverseGeocodeLocation:[[CLLocation alloc] initWithLatitude:21.123 longitude:123.345] completionHandler:^(NSArray<CLPlacemark *> * __nullable placemarks, NSError * __nullable error)
{
// 包含区,街道等信息的地标对象
CLPlacemark *placemark = [placemarks firstObject];
// 城市名称
NSString *city = placemark.locality;
// 街道名称
NSString *street = placemark.thoroughfare;
// 全称
NSString *name = placemark.name;
}];

  1. CLPlacemark 地标对象详解
    location : CLLocation 类型, 位置对象信息, 里面包含经纬度, 海拔等等
    region : CLRegion 类型, 地标对象对应的区域
    addressDictionary : NSDictionary 类型, 存放街道,省市等信息
    name : NSString 类型, 地址全称
    thoroughfare : NSString 类型, 街道名称
    locality : NSString 类型, 城市名称
    administrativeArea : NSString 类型, 省名称
    country : NSString 类型, 国家名称
  2. 测试环境
  • 必须联网
    XCode版本不限
    iOS模拟器系统版本不限
  1. 常见问题
  2. 测试无数据?
    首先, 检查是否有联网;
    其次, 如果是反地理编码,可尝试更换经纬度再次尝试, 有的经纬度没有对应信息
  3. 应用场景
  4. 一般与定位结合使用, 确定当前位置的具体地理信息
    八. 使用第三方框架进行定位(✨✨)
  5. 主要原因
    因为使用CoreLocation框架进行获取用户位置信息, 是通过代理进行回调; 而第三方框架将"代理模拟"转换成为"block模式"; 使用起来比较方便, 而且额外增加了超时时间等功能.
  6. 框架信息
    名称: locationManager
    地址: link
  7. 使用方法
    参照该框架对应的 readME
  8. 注意事项
    一般集成第三方框架到项目中, 请先确保该框架没有问题, 然后再向项目中集成

CoreLocation :用于地理定位,地理编码,区域监听等

3. 知识补充

1. 定位常识
    1) 标准定位服务(基于gps/基站/wifi定位, 具体使用哪种,苹果有自己规则) 
        > 程序关闭,就没法获取位置 
    2) 显著的位置变化定位服务(使用基站进行定位,所以必须要求设备有电话模块) 
        > 当app被完全关闭时,也可以接收到位置通知,并让app进入到后台处理 
        > 定位精度相比于上面,精度不大,所以耗电小,而且定位更新频率依据基站密度而定 

2. 应用场景
    1) 如果要求定位及时,精度较高,并且运行时间较短,可使用标准定位; 
    2) 如果长时间监控用户位置,用户移动速度比较快(例如打车软件),可使用后者 
先通过一个简单例子看一下
#import "ViewController.h"
#import <CoreLocation/CoreLocation.h>
@interface ViewController ()<CLLocationManagerDelegate>
@property(nonatomic,strong)CLLocationManager *locationM;

@end

@implementation ViewController
-(CLLocationManager *)locationM
{
    if (_locationM == nil) {
        _locationM = [[CLLocationManager alloc]init];
        _locationM.delegate = self;
        // 只在前台开始定位  修改plist文件 提醒用户
        [_locationM requestWhenInUseAuthorization];
        // 前后台都可以定位  修改plist文件 提醒用户
        // [_locationM requestAlwaysAuthorization]; 
    }
    return _locationM;
}
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    // 开始标准定位服务
    [self.locationM startUpdatingLocation];
    // 开启显著位置变化定位服务
    // [self.locationM startMonitoringSignificantLocationChanges];
}
#pragma mark CLLocationManagerDelegate代理方法
-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations
{
    // manager 位置管理者       locations 位置数组
    // 在这里拿到位置信息做一些处理,这个方法会被持续调
    NSLog(@"--开始定位--");
}

注意:
1. 获取前台定位授权[_locationM requestWhenInUseAuthorization];需要在plist文件中加入NSLocationWhenInUseUsageDescription
获取前后台定位授权[_locationM requestAlwaysAuthorization]; 需要在plist文件中加入NSLocationAlwaysUsageDescription
plist中加入的信息表示获取定位授权时显示的信息。

plist文件修改

运行程序请求用户授权时会弹出

请求用户授权

2. 开启标准定位服务使用的是GPS/WIFI定位,精确度较高,关闭应用程序就无法获取位置,而开启显著位置变化定位服务使用基站定位(必须有电话模块),当应用程序被关闭时,也可以接受到位置通知,并让app进入后台处理,但是定位精确度没有标准定位服务高,耗电少,定位更新频率依照基站密度而定,只要在基站范围内就显示基站位置,当进入另一个基站范围后更新。
如果要求定位及时,精确度高,并且运行时间短,可以使用标准定位服务。
如果长时间监控用户位置,用户移动速度较快,可使用显著位置变化定位服务

3. 代理方法didUpdateLocations会被持续调用,参数manager位置管理者 locations表示位置数组,里面按照时间先后顺序存储CLLocation对象,获取最后一个位置信息[locations lastObject]即可

必威 1Snip20160120_1.png

MapKit :用于地图展示,例如大头针,路线、覆盖层展示等

4. 测试环境:

1. XCode7.0之前版本,例如XCode6.4版本
2. 模拟器选择iOS8.0之前的版本
    * 原因 : XCode7.0(包含7.0)之后不支持iOS8.0之前的模拟器
CLLocationManager -- 关于定位属性和方法
// 判断定位功能是否可用
+ (BOOL)locationServicesEnabled
// 设置过滤单位(米)即每隔多少米定位一次
@property(assign, nonatomic) CLLocationDistance distanceFilter;  
// 设置定位精确度
@property(assign, nonatomic) CLLocationAccuracy desiredAccuracy;
CLLocationAccuracy kCLLocationAccuracyBestForNavigation  // 最适合导航的精确度
CLLocationAccuracy kCLLocationAccuracyBest; // 最好的
CLLocationAccuracy kCLLocationAccuracyNearestTenMeters; // 附近10米范围内
CLLocationAccuracy kCLLocationAccuracyHundredMeters; // 附近100米范围内
CLLocationAccuracy kCLLocationAccuracyKilometer; // 附近1000米范围内
CLLocationAccuracy kCLLocationAccuracyThreeKilometers; // 附近1000米范围内
// 开启定位
- (void)startUpdatingLocation
// 结束定位
- (void)stopUpdatingLocation;
  1. 额外设置

2个热门专业术语

5. 常见问题总结

1. 定位不到, 对应的代理方法不执行
    首先,检查运行的模拟器是否是iOS8.0之前的系统版本
    其次,检查模拟器是否设置位置数据
    第三,确保代码无问题(一般都是代理没有设置,或者位置管理器对象是局部变量,亦或是位置管理器对象没有被强引用)
    第四,绝逼是模拟器BUG, 请重置模拟器(是重置,不是重启)

CLLocationManagerDelegate -- 定位常用代理方法
// 定位成功 持续调用
-(void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations
{
    // manager : 位置管理者       
    // locations : 位置数组
    // 在这里拿到位置信息做一些处理,这个方法会被持续调
}
// 定位失败时调用
-(void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error
{
    // manager : 位置管理者       
    // error : 错误信息
}
// 当用户定位授权状态发生变化时调用
-(void)locationManager:(CLLocationManager *)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status
{
    // manager : 管理者
    // status :状态
    /*
    kCLAuthorizationStatusNotDetermined = 0 // 用户未决定时
    kCLAuthorizationStatusRestricted // 受限制保留字段
    kCLAuthorizationStatusDenied // 被拒绝 1.被拒绝 2.未开启定位服务
    kCLAuthorizationStatusAuthorizedAlways // 前后台都可以定位授权
    kCLAuthorizationStatusAuthorizedWhenInUse // 前台定位授权
    */
}

LBS :Location Based Service

二. iOS8.0之后定位

CLLocation对象 -- 定位基本属性
// 根据经度和维度创建一个CLLocation对象
- (instancetype)initWithLatitude:(CLLocationDegrees)latitude
    longitude:(CLLocationDegrees)longitude;
// 经纬度,latitude经度 longitude维度
@property(readonly, nonatomic) CLLocationCoordinate2D coordinate;
// 高度位置 可以正面(海拔)或负面(低于海平面)。
@property(readonly, nonatomic) CLLocationDistance altitude;
// 水平精确度,如果是负值表示不可用
@property(readonly, nonatomic) CLLocationAccuracy horizontalAccuracy;
// 垂直精确度,如果是负值表示海拔不可用
@property(readonly, nonatomic) CLLocationAccuracy verticalAccuracy;
// 真北的位置度 取值 0.0 - 359.9 度  0 表示真北 
@property(readonly, nonatomic) CLLocationDirection course 
// 速度 m/s 负值表示速度无效
@property(readonly, nonatomic) CLLocationSpeed speed 
// 定位时间
@property(readonly, nonatomic, copy) NSDate *timestamp;
// 楼层,如果建筑物注册,可以获取楼层
@property(readonly, nonatomic, copy, nullable) CLFloor *floor 
// 返回位置
@property (nonatomic, readonly, copy) NSString *description;
// 计算两个坐标的物理直线距离
- (CLLocationDistance)distanceFromLocation:(const CLLocation *)location 
每隔多少米定位一次 代码: self.locationM.distanceFilter = 100; 功能: 只有当最新的位置与上一次获取的位置之间的距离, 大于这个值时, 才会通过代理告诉外界.设置定位精确度 代 码: self.locationM.desiredAccuracy = kCLLocationAccuracyBest; 功 能: 通过设置此属性, 获取不同精确度的位置信息 注意事项: 精确度越高,越耗电,定位所需时间越长 枚举注解: 枚举值 含义kCLLocationAccuracyBestForNavigation 最适合导航kCLLocationAccuracyBest 精度最好的kCLLocationAccuracyNearestTenMeters 附近10米kCLLocationAccuracyHundredMeters 附近100米kCLLocationAccuracyKilometer 附近1000米kCLLocationAccuracyThreeKilometers 附近3000米

SoLoMo :Social LocalMobile

1. 前台定位

  1. 导入CoreLocation框架以及对应的主头文件

     #import <CoreLocation/CoreLocation.h>
    
  2. 创建CLLocationManager对象并设置代理

     self.locationM = [[CLLocationManager alloc] init];
     self.locationM.delegate = self;
    
  3. 请求前台定位授权, 并在Info.Plist文件中配置Key ( Nslocationwheninuseusagedescription )

     [self.locationM requestWhenInUseAuthorization];
    
  4. 调用方法,开始更新用户位置信息

     [self.locationM startUpdatingLocation];
    
  5. 在对应的代理方法中获取位置信息

     -(void)locationManager:(nonnull CLLocationManager *)manager didUpdateLocations:(nonnull NSArray<CLLocation > *)locations  
     {
         NSLog(@"每当请求到位置信息时, 都会调用此方法");  
     }
    

2. CLLocationManager -- 设备方向

  1. 定位常识

    1. 标准定位服务(基于gps/基站/wifi定位, 具体使用哪种,苹果有自己规则)

      程序关闭,就没法获取位置2) 显著的位置变化定位服务(使用基站进行定位,所以必须要求设备有电话模块)> 当app被完全关闭时,也可以接收到位置通知,并让app进入到后台处理> 定位精度相比于上面,精度不大,所以耗电小,而且定位更新频率依据基站密度而定

  2. 应用场景

    1. 如果要求定位及时,精度较高,并且运行时间较短,可使用标准定位;
    2. 如果长时间监控用户位置,用户移动速度比较快,可使用后者
  3. 常见问题总结

  4. 定位不到, 对应的代理方法不执行首先,检查运行的模拟器是否是iOS8.0之前的系统版本其次,检查模拟器是否设置位置数据第三,确保代码无问题(一般都是代理没有设置,或者位置管理器对象是局部变量,亦或是位置管理器对象没有被强引用)第四,模拟器BUG,

    必威 2Snip20160120_2.png

必威 31.png

2. 后台定位

手机通过磁力计来判断设备方向,先看一个简单指南针的例子
#import "ViewController.h"
#import <CoreLocation/CoreLocation.h>
@interface ViewController ()<CLLocationManagerDelegate>
@property (weak, nonatomic) IBOutlet UIImageView *imageView;
@property(nonatomic,strong)CLLocationManager *locationM;

@end

@implementation ViewController
-(CLLocationManager *)locationM
{
    if (_locationM == nil) {
        _locationM = [[CLLocationManager alloc]init];
        _locationM.delegate = self;
    }
    return _locationM;
}
- (void)viewDidLoad {
    [super viewDidLoad];
    [self.locationM startUpdatingHeading];
}
-(void)locationManager:(CLLocationManager *)manager didUpdateHeading:(CLHeading *)newHeading
{
    /**
     *  CLHeading
     *  magneticHeading : 距离磁北方向的角度
     *  trueHeading : 距离真北方向的角度
     * headingAccuracy : 如果这个值是负数, 那么代表角度不可用
     */
    if (newHeading.headingAccuracy < 0) {
        return;
    }
    CGFloat angle = newHeading.magneticHeading;
    CGFloat r = angle * M_PI / 180;
    [UIView animateWithDuration:0.5 animations:^{
        self.imageView.transform = CGAffineTransformMakeRotation(-r);
    }];
}

二. iOS8.0之后定位

CLLocationManager

方案一:
  1. 在前台定位基础上, 勾选后台模式Location updates
CLLocationManager -- 关于手机朝向属性和方法
// 判断是否支持磁力计定位手机朝向
+ (BOOL)headingAvailable
// 开启手机朝向定位
- (void)startUpdatingHeading 
// 关闭手机朝向定位
- (void)stopUpdatingHeading
1. 前台定位导入CoreLocation框架以及对应的主头文件 #import <CoreLocation/CoreLocation.h>创建CLLocationManager对象并设置代理 self.locationM = [[CLLocationManager alloc] init]; self.locationM.delegate = self;请求前台定位授权, 并在Info.Plist文件中配置Key ( Nslocationwheninuseusagedescription ) [self.locationM requestWhenInUseAuthorization];调用方法,开始更新用户位置信息 [self.locationM startUpdatingLocation];在对应的代理方法中获取位置信息 -locationManager:(nonnull CLLocationManager *)manager didUpdateLocations:(nonnull NSArray<CLLocation > *)locations { NSLog(@"每当请求到位置信息时, 都会调用此方法"); }

2. 后台定位方案一:在前台定位基础上, 勾选后台模式Location updates image注意:此时授权状态如果是前台定位, 那么当APP退到后台时, 屏幕顶部会出现蓝条方案二:请求前后台定位授权,并在info.plist文件中配置KEY ( NSLocationAlwaysUsageDescription ) [self.locationM requestAlwaysAuthorization];注意:不需要勾选后台模式, 也可以进行后台定位注意:此时授权状态如果是前后台定位, 那么即使APP退到后台时, 屏幕顶部会也不会出现蓝条

3. 监听用户授权状态实现以下代理方法即可 // 当用户授权状态发生变化时调用 -locationManager:(nonnull CLLocationManager *)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status { switch  { // 用户还未决定 case kCLAuthorizationStatusNotDetermined: { NSLog(@"用户还未决定"); break; } // 访问受限(苹果预留选项,暂时没用) case kCLAuthorizationStatusRestricted: { NSLog; break; } // 定位关闭时和对此APP授权为never时调用 case kCLAuthorizationStatusDenied: { // 定位是否可用(是否支持定位或者定位是否开启) if([CLLocationManager locationServicesEnabled]) { NSLog(@"定位开启,但被拒"); // 在此处, 应该提醒用户给此应用授权, 并跳转到"设置"界面让用户进行授权 // 在iOS8.0之后跳转到"设置"界面代码 NSURL *settingURL = [NSURL URLWithString:UIApplicationOpenSettingsURLString]; if([[UIApplication sharedApplication] canOpenURL:settingURL]) { [[UIApplication sharedApplication] openURL:settingURL]; } }else { NSLog(@"定位关闭,不可用"); } break; } // 获取前后台定位授权 case kCLAuthorizationStatusAuthorizedAlways: // case kCLAuthorizationStatusAuthorized: // 失效,不建议使用 { NSLog(@"获取前后台定位授权"); break; } // 获得前台定位授权 case kCLAuthorizationStatusAuthorizedWhenInUse: { NSLog(@"获得前台定位授权"); break; } default: break; } }

CLLocationManager的常用操作

注意:此时授权状态如果是前台定位, 那么当APP退到后台时, 屏幕顶部会出现蓝条
CLLocationManagerDelegaer -- 关于手机朝向的代理方法
// 当获取一个新朝向的时候调用,持续调用
- (void)locationManager:(CLLocationManager *)manager didUpdateHeading:(CLHeading *)newHeading
{
    // manager : 管理者
    // newHeading : 朝向
}

三. iOS9.0 定位补充

开始更新用户位置

方案二:
  1. 请求前后台定位授权,并在info.plist文件中配置KEY ( NSLocationAlwaysUsageDescription )

     [self.locationM requestAlwaysAuthorization];
    

    注意:不需要勾选后台模式, 也可以进行后台定位

    注意:此时授权状态如果是前后台定位, 那么即使APP退到后台时, 屏幕顶部会也不会出现蓝条

CLHeading对象基本属性
// 与磁北方向的角度  范围 0.0 - 359.9 度, 0 度表示磁北方向
@property(readonly, nonatomic) CLLocationDirection magneticHeading;
// 与真北方向的角度  范围 0.0 - 359.9 度, 0 度表示真北方向
@property(readonly, nonatomic) CLLocationDirection trueHeading;
// 返回方向值的错误范围,负值表示无效的朝向
@property(readonly, nonatomic) CLLocationDirection headingAccuracy;
// 返回方向的时间
@property(readonly, nonatomic, copy) NSDate *timestamp;

注意:当获取朝向的时候不需要向用户请求授权,因为设备方向不涉及到用户隐私

1. 定位变化* 前台定位 (同iOS8.0之后一致, 无任何变化, 都需要主动请求授权)* 后台定位方案一:在前台定位基础上, 勾选后台模式Location updates, 并且设置以下属性为YES if ([[UIDevice currentDevice].systemVersion floatValue] >= 9.0) { self.locationM.allowsBackgroundLocationUpdates = YES; }方案二:请求前后台定位授权,并在info.plist文件中配置KEY ( NSLocationAlwaysUsageDescription ) [self.locationM requestAlwaysAuthorization];

-startUpdatingLocation;

3. 监听用户授权状态

  1. 实现以下代理方法即可

     // 当用户授权状态发生变化时调用
     -(void)locationManager:(nonnull CLLocationManager *)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status
     {
     switch (status) {
     // 用户还未决定
     case kCLAuthorizationStatusNotDetermined:
     {
         NSLog(@"用户还未决定");
         break;
     }
     // 访问受限(苹果预留选项,暂时没用)
     case kCLAuthorizationStatusRestricted:
     {
         NSLog(@"访问受限");
         break;
     }
     // 定位关闭时和对此APP授权为never时调用
     case kCLAuthorizationStatusDenied:
     {
         // 定位是否可用(是否支持定位或者定位是否开启)
         if([CLLocationManager locationServicesEnabled])
         {
             NSLog(@"定位开启,但被拒");
             // 在此处, 应该提醒用户给此应用授权, 并跳转到"设置"界面让用户进行授权
             // 在iOS8.0之后跳转到"设置"界面代码
             NSURL *settingURL = [NSURL URLWithString:UIApplicationOpenSettingsURLString];
             if([[UIApplication sharedApplication] canOpenURL:settingURL])
             {
                 [[UIApplication sharedApplication] openURL:settingURL];
             }
         }else
         {
             NSLog(@"定位关闭,不可用");
         }
         break;
     }
         // 获取前后台定位授权
     case kCLAuthorizationStatusAuthorizedAlways:
     //  case kCLAuthorizationStatusAuthorized: // 失效,不建议使用
     {
         NSLog(@"获取前后台定位授权");
         break;
     }
         // 获得前台定位授权
     case kCLAuthorizationStatusAuthorizedWhenInUse:
     {
         NSLog(@"获得前台定位授权");
         break;
     }
     default:
         break;
     }
     }
    

3. CLLocationManager -- 区域监听

区域监听实例

#import "ViewController.h"
#import <CoreLocation/CoreLocation.h>
@interface ViewController ()<CLLocationManagerDelegate>
@property(nonatomic,strong)CLLocationManager *locationM;

@end

@implementation ViewController
-(CLLocationManager *)locationM
{
    if (_locationM == nil) {
        _locationM = [[CLLocationManager alloc]init];
        _locationM.delegate = self;
        [_locationM requestAlwaysAuthorization];
    }
    return _locationM;
}
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
    // 中心
    CLLocationCoordinate2D center = CLLocationCoordinate2DMake(67.666, 80.888);
    // 区域半径
    CLLocationDistance distance = 1000.0;
    CLCircularRegion *range = [[CLCircularRegion alloc]initWithCenter:center radius:distance identifier:@"ding"];
    // 用这个方法需要有位置的变化才行,从外部进来 或者出去才会有响应
    // [self.locationM startMonitoringForRegion:range];
    // 用这个方法就会先获取一次,判断是否在区域中
    [self.locationM requestStateForRegion:range];
}

-(void)locationManager:(CLLocationManager *)manager didEnterRegion:(CLRegion *)region
{
    NSLog(@"进入区域");
}
-(void)locationManager:(CLLocationManager *)manager didExitRegion:(CLRegion *)region
{
    NSLog(@"出去区域");
}
-(void)locationManager:(CLLocationManager *)manager didDetermineState:(CLRegionState)state forRegion:(CLRegion *)region
{
    NSLog(@"%@",region);
    // 获取请求指定区域状态时调用的方法
    /**
     CLRegionStateUnknown, // 不知道
     CLRegionStateInside, // 在区域内部
     CLRegionStateOutside // 在区域外部
     */
    if(state == CLRegionStateInside)
    {
        NSLog(@"在区域中");
    }else if (state == CLRegionStateOutside)
    {
        NSLog(@"在区域外面");
    }
}
@end

注意:
[self.locationM startMonitoringForRegion:range];开启区域监听,需要有位置的变化才会调用代理方法,例如位置从区域外部进入区域内部。
[self.locationM requestStateForRegion:range];程序一运行就会先确定在不在区域中,当位置发生改变时也会判断在不在区域中,是进入区域还是离开区域

  1. 新的API

停止更新用户位置

4. 测试环境:

1. XCode版本无要求
2. 模拟器选择iOS8.0之后的版本
CLLocationManager -- 关于区域间厅属性和方法
// 判断当前设备是否支持区域监听(区域类型)
+ (BOOL)isMonitoringAvailableForClass:(Class)regionClass
// 最大的区域大小,超过这个最大值后无效
@property (readonly, nonatomic) CLLocationDistance maximumRegionMonitoringDistance
// 开启一个区域的监听
- (void)startMonitoringForRegion:(CLRegion *)region 
// 请求一个区域的监听
- (void)requestStateForRegion:(CLRegion *)region 

单次定位请求;

- stopUpdatingLocation;

5. 常见问题总结

1. 定位不到, 对应的代理方法不执行
    首先,检查是否请求授权, 并设置了对应的KEY
    其次,检查模拟器是否设置位置数据
    第三,确保代码无问题(一般都是代理没有设置,或者位置管理器对象是局部变量,亦或是位置管理器对象没有被强引用)
    第四,绝逼是模拟器BUG, 请重置模拟器(是重置,不是重启)

CLLocationManagerDelegaer -- 关于区域监听的代理方法
// 进入区域时调用 manager : 管理者 region:区域
-(void)locationManager:(CLLocationManager *)manager didEnterRegion:(CLRegion *)region
{
}
-(void)locationManager:(CLLocationManager *)manager didExitRegion:(CLRegion *)region
{
}
-(void)locationManager:(CLLocationManager *)manager didDetermineState:(CLRegionState)state forRegion:(CLRegion *)region
{
       state : 状态
       CLRegionStateUnknown, // 不知道
       CLRegionStateInside, // 在区域内部
       CLRegionStateOutside // 在区域外部
}
 代 码: [self.locationM requestLocation]; 功 能: 获取一次位置信息 实现逻辑:  按照定位精确度从低到高进行排序,逐个进行定位.如果在有效时间内, 定位到了精确度最好的位置, 那么就把对应的位置通过代理告知外界.  如果获取到的位置不是精确度最高的那个,也会在定位超时后,通过代理告诉外界.

 注意事项:  必须实现代理的-locationManager:didFailWithError:方法  不能与startUpdatingLocation方法同时使用

当调用了startUpdatingLocation方法后,就开始不断地请求、刷新用户的位置,一旦请求到用户位置就会调用代理的下面方法

三. iOS9.0 定位补充

CLCircularRegion对象基本属性

CLCircularRegion是CLRegion的子类

// 创建方法 cente : 中心位置 radius : 区域半径 identidier : 唯一标示 
- (instancetype)initWithCenter:(CLLocationCoordinate2D)center
                            radius:(CLLocationDistance)radius
                        identifier:(NSString *)identifier;
// 中心位置
@property (readonly, nonatomic) CLLocationCoordinate2D center;
// 半径
@property (readonly, nonatomic) CLLocationDistance radius;

四. CLLocation对象详解

-locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations;

1. 定位变化

4. 地理编码和反地理编码

地理编码指 地址转经纬度,反地理编码指 经纬度转地址。使用CLGeocoder来获取。

1. 属性解释 > coordinate : 当前位置所在的经纬度数据 > altitude : 海拔 > speed : 当前速度 > course : 航向(设备移动的方向, 值域范围:0.0 ~ 259.9, 正北方向为0.0)2. 重要方法 代码: - (CLLocationDistance)distanceFromLocation:(CLLocation *)location 作用: 计算两个位置对象之间的物理距离, 单位是3. 场景演练1. 场景演示:打印当前用户的行走方向,偏离角度以及对应的行走距离, 例如:”北偏东30度方向,移动了8米”2. 实现步骤: 1> 获取对应的方向偏向(例如”正东”,”东偏南”) 2> 获取对应的偏离角度(并判断是否是正方向) 3> 计算行走距离 4> 打印信息4. 注意事项使用位置前, 务必判断当前获取的位置是否有效 代码: if (location.horizontalAccuracy < 0) return; 功能: 如果水平精确度小于零, 代表虽然可以获取位置对象, 但是数据错误, 不可用 

locations参数里面装着CLLocation对象

* 前台定位

    (同iOS8.0之后一致, 无任何变化, 都需要主动请求授权)
CLGeocoder 的使用
// 创建
CLGeocoder *geocoder = [[CLGeocoder alloc]init];
// 地理编码
[self.geocoder geocodeAddressString:地址 completionHandler:^(NSArray<CLPlacemark *> * _Nullable placemarks, NSError * _Nullable error) {
     if (error == nil) {
     }
}];
// 反地理编码
[self.geocoder reverseGeocodeLocation:location completionHandler:^(NSArray<CLPlacemark *> * _Nullable placemarks, NSError * _Nullable error) {
    if (error == nil) {
    }       
}];

注意:方法中返回的是一个装着CLPlacemark对象的数组,是对输入地址名称或者经纬度进行检索的结果,因此返回多个结果供选择。error指错误信息,如果错误error有值

二. tips由于定位非常耗电; 所以为了给用户省电, 你可以遵守以下小经验

CLLocationManager补充

* 后台定位

CLPlacemark基本属性
// 对应的位置对象 参考CLLocation基本属性
@property (nonatomic, readonly, copy, nullable) CLLocation *location;
@property (nonatomic, readonly, copy, nullable) CLRegion *region; // 范围
@property (nonatomic, readonly, copy, nullable) NSTimeZone *timeZone // 时区
@property (nonatomic, readonly, copy, nullable) NSString *name; // 地址名称
@property (nonatomic, readonly, copy, nullable) NSString *thoroughfare; // 街道
@property (nonatomic, readonly, copy, nullable) NSString *subThoroughfare; // 街道相关信息,例如门牌
@property (nonatomic, readonly, copy, nullable) NSString *locality; // 城市
@property (nonatomic, readonly, copy, nullable) NSString *subLocality; // 城市内分区
@property (nonatomic, readonly, copy, nullable) NSString *administrativeArea; // 直辖市
@property (nonatomic, readonly, copy, nullable) NSString *subAdministrativeArea; // 其他行政区域信息
@property (nonatomic, readonly, copy, nullable) NSString *postalCode; // 邮编
@property (nonatomic, readonly, copy, nullable) NSString *ISOcountryCode; // 国家编码
@property (nonatomic, readonly, copy, nullable) NSString *country; // 国家
@property (nonatomic, readonly, copy, nullable) NSString *inlandWater; // 水源湖泊
@property (nonatomic, readonly, copy, nullable) NSString *ocean; // 海洋
@property (nonatomic, readonly, copy, nullable) NSArray<NSString *> *areasOfInterest; // 关联的或利益相关的地标
1)不需要获取用户位置时,一定要关闭定位服务:2)如果能满足项目需求,尽可能的使用”监听显著位置变化”的定位服务如果可以,尽可能使用低精度的desiredAccuracy4)如果是数据采集,(一般都是周期性的去轮询用户位置),在轮询期间一定要关闭定位

为了严谨起见,最好在使用定位功能之前判断当前应用的定位功能是否可用

方案一:
  1. 在前台定位基础上, 勾选后台模式Location updates, 并且设置以下属性为YES

     if ([[UIDevice currentDevice].systemVersion floatValue] >= 9.0)
      {
          self.locationM.allowsBackgroundLocationUpdates = YES;
      }
    

四. iOS9/iOS8/iOS8之前的定位适配

五. 区域监听1.概念解释区 域 : 就是指划定的一块地域范围(比如圆形区域, 则由区域中心, 和半径组成)区域监听 : 是指,我们通过代码指定一个区域, 然后当用户持握设备进入或者离开指定区域, 我们都能监听到.

CLLocationManager有个类方法可以判断当前应用的定位功能是否可用

方案二:
  1. 请求前后台定位授权,并在info.plist文件中配置KEY ( NSLocationAlwaysUsageDescription )

     [self.locationM requestAlwaysAuthorization];
    

1. iOS8.0之前是默认请求授权,需要在plist文件中加入

iOS 8.0之前

设置后台执行

设置后台执行

2. 监听指定区域导入CoreLocation框架以及对应的主头文件 #import <CoreLocation/CoreLocation.h>创建CLLocationManager对象并设置代理 self.locationM = [[CLLocationManager alloc] init]; self.locationM.delegate = self;请求前后台定位, 或前台定位授权, 并在Info.Plist文件中配置相应的Key [self.locationM requestAlwaysAuthorization]; // [self.locationM requestWhenInUseAuthorization]; 创建一个区域, 并开始监听 // 1. 判断区域监听服务是否可用(定位服务是否关闭, 定位是否授权, 是否开启飞行模式) if ([CLLocationManager isMonitoringAvailableForClass:[CLCircularRegion class]]) { // 创建区域中心 CLLocationCoordinate2D center = CLLocationCoordinate2DMake(29.12345, 131.23456); // 创建区域(指定区域中心,和区域半径) CLLocationDistance radius = 1000; // 判断区域半径是否大于最大监听区域半径,如果大于, 就没法监听 if (radius > self.locationM.maximumRegionMonitoringDistance) { radius = self.locationM.maximumRegionMonitoringDistance; } CLCircularRegion *region = [[CLCircularRegion alloc] initWithCenter:center radius:radius identifier:@"小码哥"]; // 开始监听指定区域 [self.locationM startMonitoringForRegion:region]; } else { NSLog(@"区域监听不可用"); }在对应的代理方法中监听区域状态 // 进去监听区域后调用 -locationManager:(nonnull CLLocationManager *)manager didEnterRegion:(nonnull CLRegion *)region { NSLog(@"进入区域---%@", region.identifier); [manager stopMonitoringForRegion:region]; } // 离开监听区域后调用 -locationManager:(nonnull CLLocationManager *)manager didExitRegion:(nonnull CLRegion *)region { NSLog(@"离开区域---%@", region.identifier); }3. 获取某个区域的当前状态监听某个区域时, 只有进入或者离开这个区域时, 才能回调对应的方法, 是一个进入或者离开的动作 如果想知道某一个区域的当前状态(识别用户是在区域内部, 还是区域外部), 则需要使用以下方法代 码: [self.locationM requestStateForRegion:region];回调代理: // 请求某个区域状态时, 回调的代理方法 -locationManager:(CLLocationManager *)manager didDetermineState:(CLRegionState)state forRegion:(CLRegion *)region{ switch  { case CLRegionStateUnknown: NSLog; break; case CLRegionStateInside: NSLog; break; case CLRegionStateOutside: NSLog; break; default: break; }}
  • locationServicesEnabled;

2. 新的API

  1. 单次定位请求;

     代   码: [self.locationM requestLocation];
     功   能: 获取一次位置信息
     实现逻辑: 
         (1) 按照定位精确度从低到高进行排序,逐个进行定位.如果在有效时间内, 定位到了精确度最好的位置, 那么就把对应的位置通过代理告知外界.
         (2) 如果获取到的位置不是精确度最高的那个,也会在定位超时后,通过代理告诉外界.
     注意事项: 
         (1) 必须实现代理的-locationManager:didFailWithError:方法
         (2) 不能与startUpdatingLocation方法同时使用
    

2. iOS 8.0

使用[_locationM requestWhenInUseAuthorization]; 请求获取前台定位,
[_locationM requestAlwaysAuthorization];请求获取前后台定位,并修改plist文件。

iOS8.0+请求授权

iOS8.0以上也可以在Background Modes中设置后台定位,但是当后台定位的时候,会出现一个蓝条提醒用户正在后台定位

后台定位提醒

  1. 注意事项

@property(assign, nonatomic) CLLocationDistancedistanceFilter;

3. 测试环境:

1. XCode版本要求7.0版本以上
2. 模拟器选择iOS9.0之后的版本

3. iOS 9.0

iOS 9.0 与iOS8.0一样,唯一的区别在于,当在Background Modes中设置后台定位时,需要_locationM.allowsBackgroundLocationUpdates = YES;设置允许。
并且iOS 9.0中新添加了单次定位的方法[self.locationM requestLocation];只获取一次位置信息。
实现逻辑:
(1) 按照定位精确度从低到高进行排序,逐个进行定位.如果在有效时间内, 定位到了精确度最好的位置, 那么就把对应的位置通过代理告知外界.
(2) 如果获取到的位置不是精确度最高的那个,也会在定位超时后,通过代理告诉外界.
注意事项:
(1) 必须实现代理的-locationManager:didFailWithError:方法
(2) 不能与startUpdatingLocation方法同时使用

每隔多少米定位一次

4. 常见问题总结

1. 单次定位在模拟器上测试不出效果?
    答: 因为模拟器的位置是固定的, 所以无法测试出效果, 请使用真机进行测试.

五. 第三方框架LocationManager

第三方框架的使用非常简单,GitHub上已经讲解的很清晰。LocationManager是将CLLocationManager由代理向block的封装转换。CoreCLLocation使用代理,代码比较分散,第三方框架使用block来接收用户信息,并且额外增加了设置超时时间等功能,使用更简单方便易读。


文中如果有不对的地方欢迎指出。我是xx_cc,一只长大很久但还没有二够的家伙。

1. 想要做区域监听, 在iOS8.0之后, 必须请求位置授权 代码: [self.locationM requestAlwaysAuthorization]; 原因: 区域监听的原理就是获取用户的位置, 然后在判断该位置是否在制定区域内, 所以会涉及到用户隐私, 而在iOS8.0之后, 想要访问用户位置信息, 就需要主动请求授权;

2. 使用前, 先判断区域监听是否可用 代码: [CLLocationManager isMonitoringAvailableForClass:[CLCircularRegion class]]

3. 注意区域半径是否大于最大区域监听半径(如果大于, 则无法监听成功) 代码: radius > self.locationM.maximumRegionMonitoringDistance

6. 常见问题1. 区域监听, 测试没有效果? 首先, 确定代码没有问题, 是否有请求授权; 其次, 尝试修改模拟器位置信息, 触发进入区域或离开区域的动作 第三, 如果模拟器出现BUG, 定位不到, 也会无法判定当前区域状态; 所以, 最后可以尝试重置模拟器.

@property(assign, nonatomic) CLLocationAccuracydesiredAccuracy;

四. CLLocation对象详解

七. 地理编码

定位精确度

1. 属性解释

    > coordinate    : 当前位置所在的经纬度数据
    > altitude      : 海拔
    > speed         : 当前速度
    > course        : 航向(设备移动的方向, 值域范围:0.0 ~ 259.9, 正北方向为0.0)
1. 概念解释地理编码: 是指根据地址关键字, 将其转换成为对应的经纬度等信息;发地理编码: 是指根据经纬度信息, 将其转换成为对应的省市区街道等信息;

2. 地理编码导入CoreLocation框架以及对应的主头文件 #import <CoreLocation/CoreLocation.h>创建CLGeocoder对象 self.geoC = [[CLGeocoder alloc] init];根据地址关键字, 进行地理编码 // 直接根据地址进行地理编码(返回结果可能有多个,因为一个地点有重名) [self.geoC geocodeAddressString:@"广州" completionHandler:^(NSArray<CLPlacemark *> * __nullable placemarks, NSError * __nullable error) { // 包含区,街道等信息的地标对象 CLPlacemark *placemark = [placemarks firstObject]; // 城市名称 NSString *city = placemark.locality; // 街道名称 NSString *street = placemark.thoroughfare; // 全称 NSString *name = placemark.name; }]; 3. 反地理编码导入CoreLocation框架以及对应的主头文件 #import <CoreLocation/CoreLocation.h>创建CLGeocoder对象 self.geoC = [[CLGeocoder alloc] init];根据经纬度信息, 进行反地理编码 // 根据经纬度信息进行反地理编码 [self.geoC reverseGeocodeLocation:[[CLLocation alloc] initWithLatitude:21.123 longitude:123.345] completionHandler:^(NSArray<CLPlacemark *> * __nullable placemarks, NSError * __nullable error) { // 包含区,街道等信息的地标对象 CLPlacemark *placemark = [placemarks firstObject]; // 城市名称 NSString *city = placemark.locality; // 街道名称 NSString *street = placemark.thoroughfare; // 全称 NSString *name = placemark.name; }]; 4. CLPlacemark 地标对象详解 location : CLLocation 类型, 位置对象信息, 里面包含经纬度, 海拔等等 region : CLRegion 类型, 地标对象对应的区域 addressDictionary : NSDictionary 类型, 存放街道,省市等信息 name : NSString 类型, 地址全称 thoroughfare : NSString 类型, 街道名称 locality : NSString 类型, 城市名称 administrativeArea : NSString 类型, 省名称 country : NSString 类型, 国家名称

7. 应用场景1. 一般与定位结合使用, 确定当前位置的具体地理信息八. 使用第三方框架进行定位1. 主要原因 因为使用CoreLocation框架进行获取用户位置信息, 是通过代理进行回调; 而第三方框架将"代理模拟"转换成为"block模式"; 使用起来比较方便, 而且额外增加了超时时间等功能.2. 框架信息

名称: locationManager地址: [link](https://github.com/intuit/LocationManager)

CLLocation

2. 重要方法

    代码: - (CLLocationDistance)distanceFromLocation:(CLLocation *)location
    作用: 计算两个位置对象之间的物理距离, 单位是(米)

CLLocation用来表示某个位置的地理信息,比如经纬度、海拔等等

3. 场景演练

1. 场景演示:打印当前用户的行走方向,偏离角度以及对应的行走距离,
    例如:”北偏东30度方向,移动了8米”
2. 实现步骤: 
    1> 获取对应的方向偏向(例如”正东”,”东偏南”)
    2> 获取对应的偏离角度(并判断是否是正方向)
    3> 计算行走距离
    4> 打印信息

@property(readonly, nonatomic) CLLocationCoordinate2Dcoordinate;

4. 注意事项

  1. 使用位置前, 务必判断当前获取的位置是否有效

     代码: if (location.horizontalAccuracy < 0) return;
     功能: 如果水平精确度小于零, 代表虽然可以获取位置对象, 但是数据错误, 不可用 
    

经纬度

* 经验小结


@property(readonly, nonatomic) CLLocationDistancealtitude;

一. 定位的应用场景

    1) 导航 
    2) 电商APP,获取用户所在城市(需要与(反)地理编码联合使用) 
    3) 数据采集用户信息(例如,统计app使用分布)
    4) 查找周边(周边好友, 周边商家等等)

海拔

二. 开发经验

** 由于定位非常耗电; 所以为了给用户省电, 你可以遵守以下小经验 **

    1)不需要获取用户位置时,一定要关闭定位服务:
    2)如果能满足项目需求,尽可能的使用”监听显著位置变化”的定位服务(打车app)
    3)如果可以,尽可能使用低精度的desiredAccuracy
    4)如果是数据采集,(一般都是周期性的去轮询用户位置),在轮询期间一定要关闭定位

@property(readonly, nonatomic) CLLocationDirectioncourse;

五. 指南针效果实现

路线,航向(取值范围是0.0°~ 359.9°,0.0°代表真北方向)

1. 实现思路

  1. 利用"磁力计"传感器,获取设备朝向
  2. 根据设备朝向反向旋转"指南针"图片

@property(readonly, nonatomic) CLLocationSpeed speed;

2. 代码实现

  1. 获取设备朝向

     1) 导入CoreLocation框架以及对应的主头文件
         #import <CoreLocation/CoreLocation.h>
    
     2) 创建CLLocationManager对象并设置代理
         self.locationM = [[CLLocationManager alloc] init];
         self.locationM.delegate = self;
    
     3) 调用方法, 开始获取设备朝向
         [self.locationM startUpdatingHeading];
    
     4) 在对应的代理方法中获取设备朝向信息
         -(void)locationManager:(CLLocationManager *)manager didUpdateHeading:(CLHeading *)newHeading
         {
             // 旋转图片代码
         }
    
  2. 旋转图片

     // 1.判断当前的角度是否有效(如果此值小于0,代表角度无效)
     if(newHeading.headingAccuracy < 0)
     return;
    
     // 2.获取当前设备朝向(磁北方向)
     CGFloat angle = newHeading.magneticHeading;
    
     // 3.转换成为弧度
     CGFloat radian = angle / 180.0 * M_PI;
    
     // 4.带动画反向旋转指南针
     [UIView animateWithDuration:0.5 animations:^{
         self.compassView.transform = CGAffineTransformMakeRotation(-radian);
     }];
    

移动速度

3. 概念补充

    磁北角度: newHeading.magneticHeading   ------- 相对于"磁北方向"产生的角度
    真北角度: newHeading.trueHeading           ------- 相对于"真北方向"产生的角度

用- (CLLocationDistance)distanceFromLocation:(const CLLocation*)location方法可以计算2个位置之间的距离

4. 注意事项

    1. 获取设备朝向前, 先判断"磁力计"是否可用
        [CLLocationManager headingAvailable];

    2. 获取朝向前, 判断当前朝向信息是否有效
        if(newHeading.headingAccuracy < 0) return;

    3. 注意与"航向"的区别
        设备朝向是指手机的朝向; "航向"可以理解为设备的移动方向

    4. 使用"磁力计"传感器获取设备朝向, 不需要请求用户授权
        因为设备朝向不涉及用户隐私

必威 42.png必威 53.png

5.测试环境

    1. XCode版本无要求(建议:XCode7.0不需要开发者账号也可以进行真机调试)
    2. 必须要求真机设备(只有真机设备才有"磁力计"传感器) 

iOS 8.0+ 的定位适配

六. 区域监听

从iOS 8.0开始,苹果进一步加强了对用户隐私的保护。

1.概念解释

    区   域 :  就是指划定的一块地域范围(比如圆形区域, 则由区域中心, 和半径组成)
    区域监听 : 是指,我们通过代码指定一个区域, 然后当用户持握设备进入或者离开指定区域, 我们都能监听到.

当APP想访问用户的隐私信息时,系统不再自动弹出一个对话框让用户授权

2. 监听指定区域

  1. 导入CoreLocation框架以及对应的主头文件

     #import <CoreLocation/CoreLocation.h>
    
  2. 创建CLLocationManager对象并设置代理

     self.locationM = [[CLLocationManager alloc] init];
     self.locationM.delegate = self;
    
  3. 请求前后台定位, 或前台定位授权, 并在Info.Plist文件中配置相应的Key

     [self.locationM requestAlwaysAuthorization];
     // [self.locationM requestWhenInUseAuthorization]; 
    
  4. 创建一个区域, 并开始监听

     // 1. 判断区域监听服务是否可用(定位服务是否关闭, 定位是否授权, 是否开启飞行模式)
     if ([CLLocationManager isMonitoringAvailableForClass:[CLCircularRegion class]]) 
     {
    
         // 创建区域中心
         CLLocationCoordinate2D center = CLLocationCoordinate2DMake(29.12345, 131.23456);
    
         // 创建区域(指定区域中心,和区域半径)
         CLLocationDistance radius = 1000;
    
         // 判断区域半径是否大于最大监听区域半径,如果大于, 就没法监听
         if (radius > self.locationM.maximumRegionMonitoringDistance) {
             radius = self.locationM.maximumRegionMonitoringDistance;
         }
         CLCircularRegion *region = [[CLCircularRegion alloc] initWithCenter:center radius:radius identifier:@"小码哥"];
    
         // 开始监听指定区域
         [self.locationM startMonitoringForRegion:region];
     }
     else
     {
         NSLog(@"区域监听不可用");
     }
    
  5. 在对应的代理方法中监听区域状态

     // 进去监听区域后调用(调用一次)
     -(void)locationManager:(nonnull CLLocationManager *)manager didEnterRegion:(nonnull CLRegion *)region
     {
         NSLog(@"进入区域---%@", region.identifier);
         [manager stopMonitoringForRegion:region];
     }
    
     // 离开监听区域后调用(调用一次)
     -(void)locationManager:(nonnull CLLocationManager *)manager didExitRegion:(nonnull CLRegion *)region
     {
         NSLog(@"离开区域---%@", region.identifier);
     }
    

解决方案:

3. 获取某个区域的当前状态

监听某个区域时, 只有进入或者离开这个区域时, 才能回调对应的方法, 是一个进入或者离开的动作
如果想知道某一个区域的当前状态(识别用户是在区域内部, 还是区域外部), 则需要使用以下方法

代   码:
    [self.locationM requestStateForRegion:region];
回调代理: 
    // 请求某个区域状态时, 回调的代理方法
    -(void)locationManager:(CLLocationManager *)manager didDetermineState:(CLRegionState)state forRegion:(CLRegion *)region
{
    switch (state) {
        case CLRegionStateUnknown:
            NSLog(@"未知状态");
            break;
        case CLRegionStateInside:
            NSLog(@"在区域内部");
            break;
        case CLRegionStateOutside:
            NSLog(@"在区域外部");
            break;
        default:
         break;
    }
}

调用iOS 8.0的API,主动请求用户授权

4. 测试环境

XCode版本无要求
iOS模拟器版本无要求
  • requestAlwaysAuthorization 请求允许在前后台都能获取用户位置的授权

  • requestWhenInUseAuthorization 请求允许在前台获取用户位置的授权

5. 注意事项

1. 想要做区域监听, 在iOS8.0之后, 必须请求位置授权
   代码: [self.locationM requestAlwaysAuthorization];
   原因: 区域监听的原理就是获取用户的位置, 然后在判断该位置是否在制定区域内, 所以会涉及到用户隐私(位置), 而在iOS8.0之后, 想要访问用户位置信息, 就需要主动请求授权;

2. 使用前, 先判断区域监听是否可用
   代码:  [CLLocationManager isMonitoringAvailableForClass:[CLCircularRegion class]]

3. 注意区域半径是否大于最大区域监听半径(如果大于, 则无法监听成功)
   代码: radius > self.locationM.maximumRegionMonitoringDistance

务必在info.plist文件中配置对应的键值, 否则以上请求授权的方法不生效

6. 常见问题

1. 区域监听, 测试没有效果?
   首先, 确定代码没有问题, 是否有请求授权;
   其次, 尝试修改模拟器位置信息, 触发进入区域或离开区域的动作
   第三, 如果模拟器出现BUG, 定位不到, 也会无法判定当前区域状态; 所以, 最后可以尝试重置模拟器.

NSLocationAlwaysUsageDescription : 允许在前后台获取GPS的描述

七. (反)地理编码

NSLocationWhenInUseDescription : 允许在前台获取GPS的描述

1. 概念解释

地理编码:   是指根据地址关键字, 将其转换成为对应的经纬度等信息;
发地理编码:  是指根据经纬度信息, 将其转换成为对应的省市区街道等信息;

iOS 9.0 定位补充

2. 地理编码

  1. 导入CoreLocation框架以及对应的主头文件

     #import <CoreLocation/CoreLocation.h>
    
  2. 创建CLGeocoder对象

     self.geoC = [[CLGeocoder alloc] init];
    
  3. 根据地址关键字, 进行地理编码

     // 直接根据地址进行地理编码(返回结果可能有多个,因为一个地点有重名)
     [self.geoC geocodeAddressString:@"广州" completionHandler:^(NSArray<CLPlacemark *> * __nullable placemarks, NSError * __nullable error) 
     {
         // 包含区,街道等信息的地标对象
         CLPlacemark *placemark = [placemarks firstObject];
         // 城市名称
         NSString *city = placemark.locality;
         // 街道名称
         NSString *street = placemark.thoroughfare;
         // 全称
         NSString *name = placemark.name;
     }]; 
    

iOS 9.0 如果当前处于前台授权状态,默认是不可以后台获取用户位置。但可以设置以下属性为YES,就可以继续获取后台位置,但是会出现蓝条

3. 反地理编码

  1. 导入CoreLocation框架以及对应的主头文件

     #import <CoreLocation/CoreLocation.h>
    
  2. 创建CLGeocoder对象

     self.geoC = [[CLGeocoder alloc] init];
    
  3. 根据经纬度信息, 进行反地理编码

     // 根据经纬度信息进行反地理编码
     [self.geoC reverseGeocodeLocation:[[CLLocation alloc] initWithLatitude:21.123 longitude:123.345] completionHandler:^(NSArray<CLPlacemark *> * __nullable placemarks, NSError * __nullable error) 
     {
         // 包含区,街道等信息的地标对象
         CLPlacemark *placemark = [placemarks firstObject];
         // 城市名称
         NSString *city = placemark.locality;
         // 街道名称
         NSString *street = placemark.thoroughfare;
         // 全称
         NSString *name = placemark.name;
     }]; 
    

@property(assign, nonatomic) BOOLallowsBackgroundLocationUpdates

4. CLPlacemark 地标对象详解

    location            : CLLocation 类型, 位置对象信息, 里面包含经纬度, 海拔等等
    region              : CLRegion 类型, 地标对象对应的区域
    addressDictionary  : NSDictionary 类型, 存放街道,省市等信息
    name                : NSString 类型, 地址全称
    thoroughfare        : NSString 类型, 街道名称
    locality            : NSString 类型, 城市名称
    administrativeArea : NSString 类型, 省名称
    country             : NSString 类型, 国家名称

使用注意:必须设置打开对应的后台模式:location updates

5. 测试环境

*   必须联网
    XCode版本不限
    iOS模拟器系统版本不限

iOS 9.0 可以单次请求用户位置

6. 常见问题

1. 测试无数据?
   首先, 检查是否有联网;
   其次, 如果是反地理编码,可尝试更换经纬度再次尝试, 有的经纬度没有对应信息
  • requestLocation

7. 应用场景

1. 一般与定位结合使用, 确定当前位置的具体地理信息

-locationManager:(nonnull CLLocationManager*)manager didUpdateLocations:(nonnull NSArray<CLLocation *>*)locations 成功调用

八. 使用第三方框架进行定位(✨✨)

-locationManager:(nonnull CLLocationManager *)managerdidFailWithError:(nonnull NSError *)error失败调用

1. 主要原因

    因为使用CoreLocation框架进行获取用户位置信息, 是通过代理进行回调; 而第三方框架将"代理模拟"转换成为"block模式"; 使用起来比较方便, 而且额外增加了超时时间等功能.

CLLocationCoordinate2D

2. 框架信息

    名称: locationManager
    地址: [link](https://github.com/intuit/LocationManager)

CLLocationCoordinate2D是一个用来表示经纬度的结构体,定义如下

3. 使用方法

    参照该框架对应的 readME

typedef struct {

4. 注意事项

    一般集成第三方框架到项目中, 请先确保该框架没有问题, 然后再向项目中集成

 CLLocationDegreeslatitude; 纬度 CLLocationDegreeslongitude; 经度

九. 补充

  1. 代理模式到block模式的转换

     主要思想就是,先记录下外界传递过来的block, 然后在对应的代理方法里面执行这个block;
    

}CLLocationCoordinate2D;

一般用CLLocationCoordinate2DMake函数来创建CLLocationCoordinate2D

必威 64.png必威 75.png必威 86.png

CLGeocoder

使用CLGeocoder可以完成“地理编码”和“反地理编码”

地理编码:根据给定的地名,获得具体的位置信息(比如经纬度、地址的全称等)

反地理编码:根据给定的经纬度,获得具体的位置信息

地理编码方法

-geocodeAddressString:(NSString *)addressString completionHandler:(CLGeocodeCompletionHandler)completionHandler;

反地理编码方法

  • reverseGeocodeLocation:(CLLocation *)location completionHandler:(CLGeocodeCompletionHandler)completionHandler;

CLGeocodeCompletionHandler

当地理反地理编码完成时,就会调用CLGeocodeCompletionHandler

typedef void (^CLGeocodeCompletionHandler)(NSArray *placemarks, NSError *error);

这个block传递2个参数

error :当编码出错时(比如编码不出具体的信息)有值

place marks :里面装着CLPlacemark对象

CLPlacemark

CLPlacemark的字面意思是地标,封装详细的地址位置信息

@property (nonatomic, readonly) CLLocation *location;

地理位置

@property (nonatomic, readonly) CLRegion *region; 区域

@property (nonatomic, readonly) NSDictionary *addressDictionary;

详细的地址信息

@property (nonatomic, readonly) NSString *name;

地址名称

@property (nonatomic, readonly) NSString *locality; 城市

*************************笔记**************************

一. iOS8.0之前的定位

iOS8特性:

1.iOS8.0+前后台的定位授权描述

NSLocationAlwaysUsageDescription

2.iOS8.0前-定位描述

Privacy

  • Location Usage Description

3.iOS8.0+前台定位授权描述

NSLocationWhenInUseUsageDescription

本文由必威发布于必威-编程,转载请注明出处:是一个进入或者离开的动作必威,移动app能解决

TAG标签:
Ctrl+D 将本页面保存为书签,全面了解最新资讯,方便快捷。