一.定位功能
1.ios7中的定位
1.1 导入 CoreLocation框架
1.2 创建 CLLocationManager对象
注意:要用强指针指向这个对象,一般采用懒加载来创建
privatelazyvarmgr :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 把获取到的地理名称显示到界面
地理编码&反地理编码源代码
classViewController:UIViewController{// MARK:- 控件属性@IBOutletweakvaraddressTextView:UITextView!
@IBOutletweakvarlatitudeTextField:UITextField!
@IBOutletweakvarlongitudeTextField:UITextField!
// MARK:- 懒加载属性private lazyvargeocoder :CLGeocoder=CLGeocoder()}// MARK:- 地理编码extensionViewController{@IBActionfuncgeocode(){// 1.获取用户输入的地址名称guardletaddress = addressTextView.textelse{return}// 2.对地理名称进行地理编码geocoder.geocodeAddressString(address) { (placemarks : [CLPlacemark]?, error :NSError?)in// 1.错误校验iferror !=nil{print(error)return}// 2.对结果进行校验guardletplacemarks = placemarkselse{return}// 3.遍历所有的结果forplaceinplacemarks {print(place.name)// 获取地理位置guardletlocation = place.locationelse{continue}// 获取经纬度letlatitude = location.coordinate.latitudeletlongitude = location.coordinate.longitude// 将经纬度显示textField中self.latitudeTextField.text ="\(latitude)"self.longitudeTextField.text ="\(longitude)"} }}}// MARK:- 反地理编码extensionViewController{@IBActionfuncreverseGeocode(){// 1.获取用户输入的经纬度guardletlatitude = latitudeTextField.text,letlongitude = longitudeTextField.textelse{return}// 2.将经纬度转成CLLocation对象guardletlatitudeD =Double(latitude),letlongitudeD =Double(longitude)else{return}letlocation =CLLocation(latitude: latitudeD, longitude: longitudeD)// 3.反地理编码geocoder.reverseGeocodeLocation(location) { (placemarks : [CLPlacemark]?, error :NSError?)in// 1.错误校验iferror !=nil{print(error)return}// 2.对结果进行校验guardletplacemarks = placemarkselse{return}// 3.遍历结果forplaceinplacemarks { guardletaddressDict = place.addressDictionaryelse{continue} guardletaddressArray = addressDict["FormattedAddressLines"]as? [String]else{continue} guardletaddress = addressArray.lastelse{continue}self.addressTextView.text = address } }}}
六.把定位封装为工具类
1.将工具类设计成单例对象
2.封装请求方法,在方法中传入闭包
2.1 使用属性将闭包保存起来. (因为在代理方法才能拿到位置信息)
2.2 请求用户位置(1.懒加载管理者对象,并在对象中直接设置请求授权和代理)
3.在代理方法中获得用户位置信息,并赋值给闭包属性
4.停止请求用户的位置
mgr.stopUpdatingLocation()
5.当第一次发送请求位置信息,会返回多次位置信息,怎么解决这个问题?
用户只需要定义一个Bool属性,对属性进行判断,为true就接收返回的位置信息
七.使用第三方框架请求位置信息
去github搜索LocationManager 找到框架去使用
一般用oc版的,swift也能用
文/尕小天(作者)
原文链接:http://www.jianshu.com/p/20dbde944f2a
著作权归作者所有,转载请联系作者获得授权,并标注“作者”。