iOS开发之定位

在iOS开发中,定位是不少App都须要使用的功能。本文主要对iOS中的定位知识点进行介绍。本文代码环境为:Xcode 10.1 + Swift 4.2bash

1、模块与常见类

  • 定位所包含的类都在CoreLocation模块中,因此必须导入import CoreLocation
  • CLLocation:表示某个位置的地理信息,好比经纬度、海拔等
  • CLLocationManager:定位管理器,能够理解为定位不能本身工做,须要有个管理者对它进行全过程监督。
  • CLGeocoder:地理编码,分为两种
    • 正向地理编码:根据位置信息,获取具体的经纬度等信息
    • 反向地理编码:根据给定的经纬度等信息,获取位置信息
  • CLPlacemark:位置信息,包含的信息如国家、城市、街道等
  • CLLocationManagerDelegate:定位代理,不论是定位成功与失败,都会有相应的代理方法回调
  • 具体的工做流程 (1)CLLocationManager发起定位,定位成功或者失败都会回调CLLocationManagerDelegate中相应的代理方法 (2)在成功的代理方法中获取 CLLocation 对象,进而获取经纬度 (3)经过 CLGeocoder获取经纬度对应的位置信息CLPlacemark (4)经过CLPlacemark获取具体的位置信息

2、权限

在iOS中,隐私保护特别好,凡事须要定位的时候,第一次必须弹出对话框给用户选择,一共有两种权限ide

  • 使用时才定位权限,使用这种,必须走两步 (1)程序中发起 requestWhenInUseAuthorization (2)在info.plist对应的位置写明申请权限的具体缘由
  • 一直能够定位权限,使用这种,也是两步 (1)程序中发起 requestAlwaysAuthorization (2)在info.plist对应的位置写明申请权限的具体缘由
  • 配置字段说明
    • iOS 8以前只须要配置 Privacy - Location Usage Description
    • iOS 8 - iOS 10 只有两个配置 Privacy - Location Always Usage Description Privacy - Location When In Use Usage Description
    • iOS 11以后多了Privacy - Location Always and When In Use Usage Description,因此iOS11以后必须配置的是 Privacy - Location When In Use Usage DescriptionPrivacy - Location Always and When In Use Usage Description
      权限缘由填写

注意:上架的App这个缘由必须写明确测试

3、模拟器定位

因为定位须要GPS,因此通常状况下,都须要真机进行测试,笔者在教学过程当中,常用的是一种模拟定位,这种定位须要准备一个gpx 的文件,能够取名 XXX.gpx,里面的内容以下:ui

<?xml version="1.0" encoding="UTF-8" ?>
<gpx version="1.1"
     creator="GMapToGPX 6.4j - http://www.elsewhere.org/GMapToGPX/"
     xmlns="http://www.topografix.com/GPX/1/1"
     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
     xsi:schemaLocation="http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd">
      <!--安徽商贸职业技术学院 谷歌地球:31.2906511800,118.3623587000-->
      <wpt lat="31.2906511800" lon="118.3623587000">
          <name>安徽商贸职业技术学院</name>
          <cmt>中国安徽省芜湖市弋江区文昌西路24号 邮政编码: 241002</cmt>
          <desc>中国安徽省芜湖市弋江区文昌西路24号 邮政编码: 241002</desc>
      </wpt>
</gpx>
复制代码

将本身的定位信息填写进xml对应的位置便可,而后选择Edit Scheme,在Options中选择本身的gpx 的文件,这样模拟器运行的时候就会读取该文件的位置信息。 编码

Edit Scheme

4、后台定位

若是你的App须要后台定位,能够这样作,首先在Capabilities中打开后台模式 spa

后台模式
前面说过定位权限分两种,针对这两种状况,后台定位的代码不同,效果也不同

  • 使用时才定位权限须要加上locationManager.allowsBackgroundLocationUpdates = true 开启后台定位,而一直能够定位权限不须要写任何额外代码
  • 使用时才定位权限退出后,手机顶部会有蓝条提示,而一直能够定位权限则没有

后台定位蓝条提醒

5、开发步骤与示例代码

  1. 导入CoreLocation模块
  2. 建立CLLcationManager对象,设置参数和代理,请求定位受权并配置info.plist
  3. 调用CLLcationManager对象的startUpdatingLocation方法进行定位
  4. 实现代理方法,在定位成功的方法中进行位置信息的处理
import UIKit
import CoreLocation

class ViewController: UIViewController {   
    //定位须要一个CLLocationManager
    lazy var locationManager:CLLocationManager = CLLocationManager()

    override func viewDidLoad() {
        super.viewDidLoad()
        setupManager()
    }

    func setupManager(){
        
        //默认状况是这样的,每当位置改变时LocationManager就调用一次代理。经过设置distanceFilter能够实现当位置改变超出必定范围时LocationManager才调用相应的代理方法。这样能够达到省电的目的。
        locationManager.distanceFilter = 300      
        //精度 好比为10 就会尽可能达到10米之内的精度
        locationManager.desiredAccuracy = kCLLocationAccuracyBest 
        //代理
        locationManager.delegate = self 
        //第一种:能后台定位可是会在顶部出现大蓝条(打开后台定位的开关)
        //容许后台定位
        locationManager.allowsBackgroundLocationUpdates = true
        locationManager.requestWhenInUseAuthorization()   
        //第二种:能后台定位而且不会出现大蓝条
        //locationManager.requestAlwaysAuthorization()
        
    }

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        
        //发起位置更新(定位)会一直轮询,耗电
        self.locationManager.startUpdatingLocation()
    }
    
}


extension ViewController : CLLocationManagerDelegate{
    
    //定位成功
    func locationManager(_ manager: CLLocationManager, didUpdateLocations locations: [CLLocation]) {
        
        let location = locations.last      
        //地理编码的类
        let gecoder = CLGeocoder()      
        if let location = location {
            //反地理编码 转换成 具体的地址
            gecoder.reverseGeocodeLocation(location) { (placeMarks, error) in            
                //CLPlacemark -- 国家 城市 街道
                let placeMark = placeMarks?.first        
                if let placeMark = placeMark{      
                    print("\(placeMark.country!) -- \(placeMark.name!) -- \(placeMark.locality!)")
                }       
            }
        }
        self.locationManager.stopUpdatingLocation()    
    }
    
    //定位失败
    func locationManager(_ manager: CLLocationManager, didFailWithError error: Error) {
        print(error)
        
    } 
}
复制代码

6、代码运行效果

运行效果
相关文章
相关标签/搜索