IOS(swift)获取用户定位及坐标转换

前言

IOS开发中经常会需要使用定位的功能,主要使用的是CoreLocation.framework

具体使用

Capabilities中打开 定位功能 Sign And Capabilities>Background Modes>Location Updates
Info中加入权限
Privicy - Location Always Usage Description
Privicy - Location Always And When In Usage Description

在文件中引入

// 第一步 引入api
import CoreLocation

public class LocationManager: NSObject{
    // 单例方法
    static let sharedInstance: LocationManager = {
        let instance = LocationManager()
        // setup code
        return instance
    }()

    var locationManager: CLLocationManager?

}

// 调用此方法初始化定位功能
public func initialize(){
    if (self.locationManager == nil) {
       self.locationManager = CLLocationManager()
       // 设置代理 
       self.locationManager?.delegate = self
       // 设置定位精度
       locationManager?.desiredAccuracy = kCLLocationAccuracyBest
       // 设置变动幅度
       locationManager?.distanceFilter = 5.0
        // 允许后台持续使用定位功能
       locationManager?.allowsBackgroundLocationUpdates = true
        // 进入后台后不停止
       self.locationManager?.pausesLocationUpdatesAutomatically = false
    }
}

// 开始尝试获取定位
public func startRequestLocation() {
    if (self.locationManager != nil) && (CLLocationManager.authorizationStatus() == .denied) {
        // 没有获取到权限,再次请求授权
        self.locationManager?.requestWhenInUseAuthorization()
    } else {
        locationManager?.startUpdatingLocation()
    }
}

// 实现代理
extension LocationManager: CLLocationManagerDelegate {
    // 代理方法,位置更新时回调
    public func locationManager(_ manager: CLLocationManager, didUpdateLocations locations: [CLLocation]) {
        let location = locations.last ?? CLLocation.init()
        let coordinate = location.coordinate
        let latitude = coordinate.latitude;
        let longitude = coordinate.longitude;
        // TODO... 实现自己的业务
        // 注意,这里获取到的是标准坐标。WGS-84标准
    }
    
    // 代理方法,当定位授权更新时回调
    public func locationManager(_ manager: CLLocationManager, didChangeAuthorization status: CLAuthorizationStatus) {
        // CLAuthorizationStatus
        // .notDetermined   用户还没有选择授权
        // .restricted   应用没有授权用户定位
        // .denied 用户禁止定位
        // .authorizedAlways 用户授权一直可以获取定位
        // .authorizedWhenInUse 用户授权使用期间获取定位
        // TODO...
        if status == .notDetermined {
            self.startRequestLocation()
        } else if (status == .restricted) {
            // 受限制,尝试提示然后进入设置页面进行处理
            
        } else if (status == .denied) {
            // 被拒绝,尝试提示然后进入设置页面进行处理
            
        }
    }

    // 当获取定位出错时调用
    public func locationManager(_ manager: CLLocationManager, didFailWithError error: Error) {
        // 这里应该停止调用api
        self.locationManager?.stopUpdatingLocation()
    }

}

坐标转换

在各大地图平台中,存在几种标准坐标系

  • WGS-84 标准坐标系
    IOS系统定位坐标
    Google Earth坐标
  • GCJ-02 中国坐标系
    Google Map、高德地图、腾讯地图
  • BD-09 百度坐标系
    百度坐标偏移标准,Baidu Map使用

具体转换方法为

//
//  LocationUtil.swift


import Foundation
import CoreLocation

// 参考来源:
// http://blog.woodbunny.com/post-68.html
// https://www.jianshu.com/p/347e4dc3d05a
// 圆周率
// let pi = 3.14159265358979324;
// let pi = M_PI;
let pi = Double.pi
let xpi: Double = pi * 3000.0 / 180.0

// 地球的平均半径
let r = 6371004
let a: Double = 6378245.0
// 
let e: Double = 0.00669342162296594323



//WGS-84:是国际标准,GPS坐标(Google Earth使用、或者GPS模块)
//GCJ-02:中国坐标偏移标准,Google Map、高德、腾讯使用
//BD-09: 百度坐标偏移标准,Baidu Map使用
class LocationUtil
{
    // 坐标转换 标准坐标系-> 中国坐标系
    //          WGS-84 --> GCJ-02
    // wgsLocation:     标准坐标
    public static func transformWGSToGCJ(wgsLocation:CLLocationCoordinate2D)->CLLocationCoordinate2D
    {
        var adjustLocation  = CLLocationCoordinate2D()
        var adjustLatitude  = transformLatitudeWith(x: wgsLocation.longitude - 105.0,
                                                   y:wgsLocation.latitude - 35.0);
        var adjustLongitude = transformLongitudeWith(x: wgsLocation.longitude - 105.0,
                                                     y:wgsLocation.latitude - 35.0);
        let radLatitude     = wgsLocation.latitude / 180.0 * pi;
        var magic           = sin(radLatitude);
        magic               = 1 - e * magic * magic;
        let sqrtMagic       = sqrt(magic);
        adjustLatitude      = (adjustLatitude * 180.0) / ((a * (1 - e)) / (magic * sqrtMagic) * pi);
        adjustLongitude     = (adjustLongitude * 180.0) / (a / sqrtMagic * cos(radLatitude) * pi);
        
        adjustLocation.latitude     = wgsLocation.latitude + adjustLatitude;
        adjustLocation.longitude    = wgsLocation.longitude + adjustLongitude;
        return adjustLocation;
    }
    
    // 纬度转换
    //
    public static  func transformLatitudeWith(x: Double,
                                          y: Double ) -> Double
    {
        var lat = -100.0 + 2.0 * x + 3.0 * y + 0.2 * y * y + 0.1 * x * y ;
        lat += 0.2 * sqrt(fabs(x));
        
        lat += (20.0 * sin(6.0 * x * pi)) * 2.0 / 3.0;
        lat += (20.0 * sin(2.0 * x * pi)) * 2.0 / 3.0;
        lat += (20.0 * sin(y * pi)) * 2.0 / 3.0;
        lat += (40.0 * sin(y / 3.0 * pi)) * 2.0 / 3.0;
        lat += (160.0 * sin(y / 12.0 * pi)) * 2.0 / 3.0;
        lat += (320 * sin(y * pi / 30.0)) * 2.0 / 3.0;
        return lat;
    }
    
    // 经度转换
    //
    public static func transformLongitudeWith(x: Double,
                                              y: Double )-> Double
    {
        var lon = 300.0 + x + 2.0 * y + 0.1 * x * x + 0.1 * x * y ;
        lon +=  0.1 * sqrt(fabs(x));
        lon += (20.0 * sin(6.0 * x * pi)) * 2.0 / 3.0;
        lon += (20.0 * sin(2.0 * x * pi)) * 2.0 / 3.0;
        lon += (20.0 * sin(x * pi)) * 2.0 / 3.0;
        lon += (40.0 * sin(x / 3.0 * pi)) * 2.0 / 3.0;
        lon += (150.0 * sin(x / 12.0 * pi)) * 2.0 / 3.0;
        lon += (300.0 * sin(x / 30.0 * pi)) * 2.0 / 3.0;
        return lon;
    }
    
    // 坐标转换 中国坐标系 -> 百度坐标系
    //          GCJ-02 --> BD-09
    // location:     中国标准坐标
    public static func transformGCJToBaidu(location: CLLocationCoordinate2D) -> CLLocationCoordinate2D
    {
        let z = sqrt(location.longitude * location.longitude + location.latitude * location.latitude)
            + 0.00002 * sqrt(location.latitude * pi);
        let t = atan2(location.latitude, location.longitude) + 0.000003 * cos(location.longitude * pi);
        var geoPoint        = CLLocationCoordinate2D();
        geoPoint.latitude   = (z * sin(t) + 0.006);
        geoPoint.longitude  = (z * cos(t) + 0.0065);
        return geoPoint;
    }

    // 坐标转换 百度坐标系 -> 中国坐标系
    //          BD-09 --> GCJ-02
    // location:     百度坐标

    //
    public static func transformBaiduToGCJ(location: CLLocationCoordinate2D)-> CLLocationCoordinate2D
    {
        let x       = location.longitude - 0.0065;
        let y       = location.latitude - 0.006;
        let z       = sqrt(x * x + y * y) - 0.00002 * sin(y * xpi);
        let t       = atan2(y, x) - 0.000003 * cos(x *  xpi);
        var geoPoint = CLLocationCoordinate2D();
        geoPoint.latitude  = z * sin(t);
        geoPoint.longitude = z * cos(t);
        return geoPoint;
    }
    
    // 坐标转换 中国坐标系-> 标准坐标系
    //          GCJ-02 --> WGS-84
    // wgsLocation:     中国坐标
    public static func transformGCJToWGS(location: CLLocationCoordinate2D) -> CLLocationCoordinate2D
    {
        let threshold       = 0.00001;
        // The boundary
        var minLat          = location.latitude - 0.5;
        var maxLat          = location.latitude + 0.5;
        var minLng          = location.longitude - 0.5;
        var maxLng          = location.longitude + 0.5;
        
        var delta           = 1.0;
        let maxIteration    = 30;
        // Binary search
        while(true)
        {
            let leftBottom  = transformWGSToGCJ(wgsLocation: CLLocationCoordinate2D(latitude: minLat,
                                                                                    longitude: minLng));
            let rightBottom = transformWGSToGCJ(wgsLocation: CLLocationCoordinate2D(latitude: minLat,
                                                                                    longitude : maxLng));
            let leftUp      = transformWGSToGCJ(wgsLocation: CLLocationCoordinate2D(latitude: maxLat,
                                                                                    longitude: minLng));
            let midPoint    = transformWGSToGCJ(wgsLocation: CLLocationCoordinate2D(latitude: ((minLat + maxLat) / 2),
                                                                                   longitude: ((minLng + maxLng) / 2)));
            delta = fabs(midPoint.latitude - location.latitude) + fabs(midPoint.longitude - location.longitude);
            
            if(maxIteration <= 1 || delta <= threshold)
            {
                return CLLocationCoordinate2D(latitude: (minLat + maxLat) / 2, longitude: (minLng + maxLng) / 2);
                
            }
            
            if(isContains(target: location, point1: leftBottom, point2: midPoint))
            {
                maxLat = (minLat + maxLat) / 2;
                maxLng = (minLng + maxLng) / 2;
            }
            else if(isContains(target: location, point1: rightBottom, point2: midPoint))
            {
                maxLat = (minLat + maxLat) / 2;
                minLng = (minLng + maxLng) / 2;
            }
            else if(isContains(target: location, point1: leftUp, point2: midPoint))
            {
                minLat = (minLat + maxLat) / 2;
                maxLng = (minLng + maxLng) / 2;
            }
            else
            {
                minLat = (minLat + maxLat) / 2;
                minLng = (minLng + maxLng) / 2;
            }
        }
        
    }
    
    //WGS-84 --> BD-09
    public static func transformFromWGSToBaidu(location: CLLocationCoordinate2D) -> CLLocationCoordinate2D
    {
        let gcjLocation = transformWGSToGCJ(wgsLocation: location);
        let bdLocation = transformGCJToBaidu(location: gcjLocation)
        return bdLocation;
    }
    
    //BD-09 --> WGS-84
    public static func transformFromBaiduToWGS(location: CLLocationCoordinate2D) -> CLLocationCoordinate2D
    {
        let gcjLocation = transformBaiduToGCJ(location: location);
        let wgsLocation = transformGCJToWGS(location: gcjLocation);
        return wgsLocation;
    }
    
    //判断点是否在p1和p2之间
    //point: 点
    //point1:    点1
    //point2:    点2
    public static func isContains(target:CLLocationCoordinate2D ,
                                  point1:CLLocationCoordinate2D,
                                  point2: CLLocationCoordinate2D ) -> Bool
    {
        let latitudeIn = target.latitude >= min(point1.latitude, point2.latitude) && target.latitude <= max(point1.latitude, point2.latitude);
        let longitudeIn = target.longitude >= min(point1.longitude,point2.longitude) && target.longitude <= max(point1.longitude, point2.longitude);
        
        return latitudeIn && longitudeIn;
    }
    
    public static func isLocationOutOfChina(location:CLLocationCoordinate2D) -> Bool
    {
        if (location.longitude < 72.004 || location.longitude > 137.8347 || location.latitude < 0.8293 || location.latitude > 55.8271){
            return true;
        }else{
            return false;
        }
    }

    
    
    
    // 获取两点之间的距离(米)
    public static func distanceBeteen(point1Latitude    : Double,
                                point1Longitude   : Double,
                                point2Latitude    : Double,
                                point2Longitude   : Double )->Double{
        let dd          = pi / 180;
        let x1          = point1Latitude * dd;
        let x2          = point2Latitude * dd;
        let y1          = point1Longitude * dd;
        let y2          = point2Longitude * dd;
    
        let t = 2 - 2 * cos(x1) * cos(x2) * cos(y1 - y2) - 2 * sin(x1) * sin(x2);
        let distance = Double(2) * Double(r) * asin(sqrt(t) / 2);
        return   distance;
        
    }
    
    // 获取两点之间的距离 (米)
    public static func distanceBeteen(point1:CLLocationCoordinate2D,
                                 point2:CLLocationCoordinate2D ) -> Double{
        let distance = distanceBeteen(point1Latitude: point1.latitude,
                                      point1Longitude: point1.longitude,
                                      point2Latitude: point2.latitude,
                                      point2Longitude: point2.longitude);
        return distance;
    
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351