一.定位功能
1.ios7中的定位
1.1 导入 CoreLocation框架
1.2 创建 CLLocationManager对象
注意:要用强指针指向这个对象,一般采用懒加载来创建
private lazy var mgr : CLLocationManager = CLLocationManager()
1.3 设置代理,实现代理方法
1.4 开始定位
mgr.startUpdatingLocation()
1.5 优点:不需要设置用户的授权
在info.plist加上一个key Privacy - Location Usage Description 写在value上的文字,可以显示在提示权限的文本框内
2.ios8(之后)的定位
2.1 请求定位步骤
2.11 导入CoreLocation框架
2.12 懒加载CLLocationManager对象
2.13 请求授权 (1) whenInUse (2) always
2.14 注意:必须把授权对应的key值 添加到info.plist文件中
2.15 设置代理,实现代理方法
2.16 开始定位
2.2 定位属性的应用
2.21 精确度的使用
desiredAccuracy精确度越高,越耗电
属性接收double类型的值,不过最好传系统给定好的值
kCLLocationAccuracyBestForNavigation: 导航精确度(最精确)
kCLLocationAccuracyBest: 最好精确度(默认)
kCLLocationAccuracyNearestTenMeters: 10米的误差
kCLLocationAccuracyHundredMeters: 100米的误差
kCLLocationAccuracyKilometer: 千米误差
kCLLocationAccuracyThreeKilometers: 三千米的误差
mgr.desiredAccuracy = kCLLocationAccuracyBestForNavigation
2.22 移动一段距离,再次重新定位
设置用于移动多少距离,重新进行定位
mgr.distanceFilter = 100
2.3 位置信息的获取
2.31 发送完请求定位,怎么获取位置信息?
在代理方法的闭包中,有一个数组,返回了很多信息在里面
2.32 我们常用的信息就是经纬度
二.计算两个经纬度的距离
1.获取当前位置信息
1.1 导入框架
1.2 懒加载管理者对象
1.3 请求授权
1.4 添加key值
1.5 设置代理,实现代理方法
1.6 开始定位
2.获取另一个位置的经纬度
.
3.计算两个位置的距离 distanceFromLocation
三.简易指南针的制作
1.实现思路
监听手机头方向的改变,在手机屏幕上放一张图片,始终指向北(根据手机方向的改变旋转)
2.界面搭建
拖一个UIImageView放在屏幕中央,里面放一张图片
3.监听手机头方向的改变
3.1 怎么监听?
通过发送请求(定位服务),获取手机头的方向进行监听
3.2 具体实现
3.21 导入CoreLocation框架
3.22 懒加载CLLocationManager对象
3.23 请求授权 (1) whenInUse (2) always
3.24 注意:必须把授权对应的key值 添加到info.plist文件中
3.25 设置代理,实现代理方法
3.26 请求手机头方向
3.27 获取真北方向
3.28 将真北方向转换为弧度
3.29 让图片根据弧度进行旋转(注意:图片旋转的弧度要取反 , 要和屏幕旋转方向相反才能保持一直指向一个方向)
4.对指南针优化
4.1 真实的指南针指向一个位置,会来回摆动两下才固定位置
代码实现的指南针没有这个效果
4.2 如何实现这个效果?
可以通过一个动画来实现
// Damping : 阻力系数 (0~1.0) initialSpringVelocity:回弹速度
UIView.animateWithDuration(0.5, delay: 0.0,
usingSpringWithDamping: 0.8, initialSpringVelocity: 5.0,
options: [], animations: {
self.imageView.transform = transform
}, completion: nil)
四.区域监听
1.需求,当进入某指向区域,提醒用户进入该区域,离开该区域也对用户进行提醒
.
2.实现方案步骤
2.1 懒加载 CLLocationManager对象
2.2 请求授权(注意:必须使用alyays授权方式) ,配置info文件
2.3 设置代理
2.4 创建监听区域
2.5 实现代理方法 并 开始监听
3.注意点
3.1 当之前添加过监听区域时,再次添加新的监听区域,还会对之前的区域进行监听
3.2 不想监听之前的区域,必须通过代码移除之前的区域
五.地理编码&反地理编码
需求:输入地理名称,地理编码获得该位置的经纬度. 输入经纬度,输出对应位置的地理名称
1.界面搭建
1.1 整个界面放在屏幕的中心,怎么实现?
1.11 可以用view包装
优缺点:要做大量的约束 , 但可以应用于任何版本
1.12 ios9之后可以用UIStackView来包装
优缺点:布局简便 只能适用于ios9(之后)
2.地理编码
2.1 拿到用户输入的地理名称 (导入框架CoreLocation)
2.2 地理编码
2.21 创建 CLGeocoder对象
2.22 对地理名称进行地理编码
geocoder.geocodeAddressString(address) { (<#[CLPlacemark]?#>, <#NSError?#>) in
<#code#>
}
2.23 对闭包中的CLPlacemark数据就行解析(遍历)
一个地理名称可能对应多个地方,所以编码后的到的结果是一个数组
2.24 获取数组中元素的地理位置(经纬度)
2.25 将经纬度显示到界面
3.反地理编码
3.1 拿到用户输入的经纬度
3.2 对经纬度进行反地理编码
3.21 创建 CLGeocoder对象
3.22 对经纬度进行反地理编码
geocoder.reverseGeocodeLocation(location) { (<#[CLPlacemark]?#>, <#NSError?#>) in
<#code#>
}
3.23 对闭包中的CLPlacemark数据就行解析(遍历)
一个经纬度可能对应多个位置(苹果这么设计的) 所以编码后返回一个数组
一个位置包含多个信息(省/市/街道/国家/经纬度/) 编码后的结果是字典数组
3.24 取出数组中的一个位置(字典),再获取位置信息(取出字典的元素)
3.25 把获取到的地理名称显示到界面
地理编码&反地理编码源代码
class ViewController: UIViewController {
// MARK:- 控件属性
@IBOutlet weak var addressTextView: UITextView!
@IBOutlet weak var latitudeTextField: UITextField!
@IBOutlet weak var longitudeTextField: UITextField!
// MARK:- 懒加载属性
private lazy var geocoder : CLGeocoder = CLGeocoder()
}
// MARK:- 地理编码
extension ViewController {
@IBAction func geocode() {
// 1.获取用户输入的地址名称
guard let address = addressTextView.text else {
return
}
// 2.对地理名称进行地理编码
geocoder.geocodeAddressString(address) { (placemarks : [CLPlacemark]?, error : NSError?) in
// 1.错误校验
if error != nil {
print(error)
return
}
// 2.对结果进行校验
guard let placemarks = placemarks else {
return
}
// 3.遍历所有的结果
for place in placemarks {
print(place.name)
// 获取地理位置
guard let location = place.location else {
continue
}
// 获取经纬度
let latitude = location.coordinate.latitude
let longitude = location.coordinate.longitude
// 将经纬度显示textField中
self.latitudeTextField.text = "\(latitude)"
self.longitudeTextField.text = "\(longitude)"
}
}
}
}
// MARK:- 反地理编码
extension ViewController {
@IBAction func reverseGeocode() {
// 1.获取用户输入的经纬度
guard let latitude = latitudeTextField.text, let longitude = longitudeTextField.text else {
return
}
// 2.将经纬度转成CLLocation对象
guard let latitudeD = Double(latitude), let longitudeD = Double(longitude) else {
return
}
let location = CLLocation(latitude: latitudeD, longitude: longitudeD)
// 3.反地理编码
geocoder.reverseGeocodeLocation(location) { (placemarks : [CLPlacemark]?, error : NSError?) in
// 1.错误校验
if error != nil {
print(error)
return
}
// 2.对结果进行校验
guard let placemarks = placemarks else {
return
}
// 3.遍历结果
for place in placemarks {
guard let addressDict = place.addressDictionary else {
continue
}
guard let addressArray = addressDict["FormattedAddressLines"] as? [String] else {
continue
}
guard let address = addressArray.last else {
continue
}
self.addressTextView.text = address
}
}
}
}
六.把定位封装为工具类
1.将工具类设计成单例对象
2.封装请求方法,在方法中传入闭包
2.1 使用属性将闭包保存起来. (因为在代理方法才能拿到位置信息)
2.2 请求用户位置(1.懒加载管理者对象,并在对象中直接设置请求授权和代理)
3.在代理方法中获得用户位置信息,并赋值给闭包属性
4.停止请求用户的位置
mgr.stopUpdatingLocation()
5.当第一次发送请求位置信息,会返回多次位置信息,怎么解决这个问题?
用户只需要定义一个Bool属性,对属性进行判断,为true就接收返回的位置信息
七.使用第三方框架请求位置信息
去github搜索LocationManager 找到框架去使用
一般用oc版的,swift也能用