(iOS)会使用定位功能吗? 不会就来看看吧

一.定位功能

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 我们常用的信息就是经纬度

(iOS)会使用定位功能吗? 不会就来看看吧_第1张图片

二.计算两个经纬度的距离

1.获取当前位置信息

1.1 导入框架
1.2 懒加载管理者对象
1.3 请求授权
1.4 添加key值
1.5 设置代理,实现代理方法
1.6 开始定位

2.获取另一个位置的经纬度

.

3.计算两个位置的距离 distanceFromLocation

(iOS)会使用定位功能吗? 不会就来看看吧_第2张图片

三.简易指南针的制作

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 让图片根据弧度进行旋转(注意:图片旋转的弧度要取反 ,  要和屏幕旋转方向相反才能保持一直指向一个方向)
(iOS)会使用定位功能吗? 不会就来看看吧_第3张图片

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 实现代理方法 并 开始监听

(iOS)会使用定位功能吗? 不会就来看看吧_第4张图片

3.注意点

3.1 当之前添加过监听区域时,再次添加新的监听区域,还会对之前的区域进行监听

3.2 不想监听之前的区域,必须通过代码移除之前的区域

五.地理编码&反地理编码

需求:输入地理名称,地理编码获得该位置的经纬度. 输入经纬度,输出对应位置的地理名称

1.界面搭建

1.1 整个界面放在屏幕的中心,怎么实现?

1.11 可以用view包装
优缺点:要做大量的约束 , 但可以应用于任何版本

1.12 ios9之后可以用UIStackView来包装
优缺点:布局简便 只能适用于ios9(之后)

(iOS)会使用定位功能吗? 不会就来看看吧_第5张图片

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也能用

你可能感兴趣的:((iOS)会使用定位功能吗? 不会就来看看吧)