iOS中的定位功能
CoreLocation框架(CoreLocation.framework)可用于定位设备当前的经纬度,通过该框架,应用程序可通过附近的蜂窝基站、WIFI信号或者GPS等信息计算用户位置。
iOS SDK提供了CLLocationManager、CLLocationManagerDelegate来处理设备的定位信息,包括获取设备的方向以及进行方向检测等。
iOS开发者使用CoreLocation.framework框架进行定位非常简单,CoreLocation框架的常用API主要有如下几个:
- CLLocationManager:定位管理器类;
- CLLocationManagerDelegate:该协议代表定位管理器的delegate协议。实现该协议的对象可负责处理CLLocationManager的定位事件;
- CLLocation:该对象代表位置。该对象包含了当前设备的经度、纬度、高度、速度、路线等信息,还包含了该定位信息的水平精确度、垂直精确度以及时间戳信息;
- CLHeading:该对象代表设备的移动方向;
- CLRegion:该对象代表一个区域。一般程序不会直接使用该类,而是使用它的两个子类,即CLCircularRegion(圆形区域)和CLBeaconRegion(蓝牙信号区);
除此之外,CoreLocation框架还涉及一个CLLocationCoordinate2D结构体变量,该结构体变量包含经度、纬度两个值,其中CLLocation对象的coordinate属性就是一个CLLocationCoordinate2D结构体变量。
通过上述API不难看出,通过定位我们可以做什么?
- 获取用户当前位置,记录路线轨迹等
- 监控行车速度和行车距离
- 方向监测
- 指南针
- 区域监听,比如用户进入某个区域时推广某些信息,考勤打卡等
1.定位常用属性和方法
1.0请求定位步骤
- 导入CoreLocation框架
- 懒加载CLLocationManager对象
- 请求授权 (1) 使用期间whenInUse (2) 始终允许always
- 注意:必须把授权对应的key值 添加到info.plist文件中
- 设置代理,实现代理方法
- 开始定位
1.1获取定位权限
定位属于用户的隐私权限,所以当APP需要使用这一权限的时候,需要获取用户的授权.
在获取授权之前,我们必须把授权对应的key值 添加到info.plist文件中,否则可能会引起APP crash或者审核不通过.
跟定位相关的隐私权限主要有两种,
- NSLocationWhenInUseUsageDescription(使用期间)
- NSLocationAlwaysUsageDescription(始终允许)
通俗的说,如果只是一次性定位,不需要持续定位的话,这俩实际并无区别.
而持续定位时这俩的差别,会单独在后面讲后台持续定位的时候提及.
1.2定位属性的应用
精确度desiredAccuracy
desiredAccuracy精确度越高,越耗电,属性接收double类型的值,也可以传系统给定好的值
系统给出的枚举类型有以下:
- kCLLocationAccuracyBestForNavigation: 导航精确度(最精确)
- kCLLocationAccuracyBest: 最好精确度(默认)
- kCLLocationAccuracyNearestTenMeters: 10米的误差
- kCLLocationAccuracyHundredMeters: 100米的误差
- kCLLocationAccuracyKilometer: 千米误差
- kCLLocationAccuracyThreeKilometers: 三千米的误差
距离过滤distanceFilter
设置用于移动多少距离,重新进行定位
枚举类型同精确度,也是一个float值.
1.3位置信息的获取
通过代理方法,我们可以获取到更新后的位置信息.以下是位置具体属性的图示.
1.3后台持续定位
之前产品提了一个需求,想持续上报司导的位置信息,这就需要用到我们的后台持续定位功能了.
使用后台定位需要以下设置:
- 选中target-->Gapability,打开Background Modes模式,并勾选Location updates,如下图所示:
- 设置允许后台更新位置的属性.需注意的是,该属性必须和上一条设置配合使用,否则会crash
if (SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(@"9.0")) {
// 该模式是抵抗程序在后台被杀,申明不能够被暂停
self.locationManager.allowsBackgroundLocationUpdates = YES;
}
self.locationManager.pausesLocationUpdatesAutomatically = NO;
为什么要进行以上设置呢?
因为iOS系统下,当APP进入后台时,过三分钟之后,APP会自动进入挂起模式,也就是说如果不做任何处理的情况下,定位会被系统终止掉.当用户再进入APP的时候,定位又重新开始.所以才说APP使用定位的授权选项,使用应用期间和始终其实区别不大.如果不想被终止,就得通过后台模式设置,向苹果声明,APP在后台需要操作的权限,这样定位功能才得以保留.
使用定位的授权选项,使用应用期间和始终,唯一的区别就是,当APP退到后台并持续定位时,使用应用期间的权限下,APP顶部状态栏会有个蓝色的条,提示用户当前APP一直在使用定位.而始终这个选项则不会.
说到这里,之前GAPP按照以上设置,功能正常,但是在提交appstore的时候被拒绝了,理由是,审核人员不认为我们的APP需要持续定位.后来在笔者申诉加重新提审的情况下,苹果体谅的给我们通过了.
持续定位优化方案,可以从以下几个维度考虑:
- 精度
- 过滤距离范围
- 定时
- 重大位置变化比如方向等
耗电测试:
1.位置相对固定时,自动更新位置和交替精度(精度非常精确和宽泛时定时切换)的对比实验;
2.位置更新比较频繁情况下,二者的对比
第一种情况下,自动更新位置能坚持8个小时左右,交替精度大概在6小时;
第二种情况下,由于位置变动时间只在一个小时左右,差别不太明显;
针对GAPP产品提出想要定时持续更新司导位置的需求,最终采取了第二种方案.
2.计算经纬度之间的距离
计算经纬度之间的距离所需步骤:
- 获取当前位置信息
- 获取另一个位置的经纬度
- 计算两个位置的距离
3.地理编码&反地理编码
地理编码:输入地理名称,地理编码获得该位置的经纬度.
反地理编码:输入经纬度,输出对应位置的地理名称
地理编码步骤
- 创建 CLGeocoder对象
- 对地理名称进行地理编码
- 对闭包中的CLPlacemark数据就行解析(遍历)
一个地理名称可能对应多个地方,所以编码后的到的结果是一个数组 - 获取数组中元素的地理位置(经纬度)
- 将经纬度显示到界面
反地理编码步骤
- 拿到用户输入的经纬度
- 对经纬度进行反地理编码
- 对闭包中的CLPlacemark数据就行解析(遍历)
一个经纬度可能对应多个位置(苹果这么设计的) 所以编码后返回一个数组
一个位置包含多个信息(省/市/街道/国家/经纬度/) 编码后的结果是字典数组 - 取出数组中的一个位置(字典),再获取位置信息(取出字典的元素)
- 把获取到的地理名称显示到界面
关于地理编码和反地理编码的实例,放到区域监听的demo里做一并说明.
4.区域监听 region
当用户进入或者离开某个区域的时候,对应进行某些操作
实例:打卡
笔者仿照钉钉写的打卡的小demo,大概功能如下:
1.当不在考勤范围内时,打卡显示外勤打卡,颜色区分显示,并提示当前打卡位置
2.当位置更换,进入考勤范围时,颜色改变,打卡显示正常打卡,并提示当前打卡位置
4.1区域监听步骤:
- 确定区域中心坐标点
- 确定区域半径
- 区域监听方法,离开某个区域和进入某个区域时做相应处理
4.2注意点
- 当之前添加过监听区域时,再次添加新的监听区域,还会对之前的区域进行监听
- 不想监听之前的区域,必须通过代码移除之前的区域
- 使用前先判断区域监听是否可用
- 注意监听区域的个数 (区域监听个数有上限)
- 注意区域半径是否大于最大监听半径