中国国测局地理坐标(GCJ-02)、世界标准地理坐标(WGS-84) 、百度地理坐标(BD-09)坐标系转换

YJLocationConverter

中国国测局地理坐标(GCJ-02)<火星坐标>、世界标准地理坐标(WGS-84) 、百度地理坐标(BD-09)坐标系转换工具类

需求

一、在进行地图开发过程中,我们一般能接触到以下三种类型的地图坐标系:

  • 1.WGS-84原始坐标系,一般用国际GPS纪录仪记录下来的经纬度,通过GPS定位拿到的原始经纬度,Google和高德地图定位的的经纬度(国外)都是基于WGS-84坐标系的;但是在国内是不允许直接用WGS84坐标系标注的,必须经过加密后才能使用;

  • 2.GCJ-02坐标系,又名“火星坐标系”,是我国国测局独创的坐标体系,由WGS-84加密而成,在国内,必须至少使用GCJ-02坐标系,或者使用在GCJ-02加密后再进行加密的坐标系,如百度坐标系。高德和Google在国内都是使用GCJ-02坐标系,可以说GCJ-02是国内最广泛使用的坐标系;

  • 3.百度坐标系:bd-09,百度坐标系是在GCJ-02坐标系的基础上再次加密偏移后形成的坐标系,只适用于百度地图。(目前百度API提供了从其它坐标系转换为百度坐标系的API,但却没有从百度坐标系转为其他坐标系的API);

  • 4.LocationManager 中所用到的是国际标准的坐标系统(WGS-84),使用 LocationManager 必须经过装换才能准确;

  • 5.在MKMapView上通过定位自己位置所获得的经纬度有是准确,由此可知,Apple已经对国内地图做了偏移优化.

Installation

Cocoapods

YJBannerView is available through CocoaPods. To install it, simply add the following line to your Podfile:

    pod 'YJLocationConverter'

Support-Method

/**
 世界标准地理坐标(WGS-84) 转换成 中国国测局地理坐标(GCJ-02)<火星坐标> 超出中国依旧是世界标准地理坐标

 @param location 世界标准地理坐标(WGS-84)
 @return 中国国测局地理坐标(GCJ-02)<火星坐标>
 */
+ (CLLocationCoordinate2D)yj_WGS84ConvertToGCJ02:(CLLocationCoordinate2D)location;

/**
 中国国测局地理坐标(GCJ-02)<火星坐标>  转换成  世界标准地理坐标(WGS-84) 超出中国依旧是世界标准地理坐标

 @param location 中国国测局地理坐标(GCJ-02)<火星坐标>
 @return 世界标准地理坐标(WGS-84)
 */
+ (CLLocationCoordinate2D)yj_GCJ02ConvertToWGS84:(CLLocationCoordinate2D)location;

/**
 世界标准地理坐标(WGS-84) 转换成 百度地理坐标(BD-09)

 @param location 世界标准地理坐标(WGS-84)
 @return 百度地理坐标(BD-09)
 */
+ (CLLocationCoordinate2D)yj_WGS84ConvertToBD09:(CLLocationCoordinate2D)location;

/**
 中国国测局地理坐标(GCJ-02)<火星坐标> 转换成 百度地理坐标(BD-09)

 @param location  中国国测局地理坐标(GCJ-02)<火星坐标>
 @return 百度地理坐标(BD-09)
 */
+ (CLLocationCoordinate2D)yj_GCJ02ConvertToBD09:(CLLocationCoordinate2D)location;

/**
 百度地理坐标(BD-09) 转换成 中国国测局地理坐标(GCJ-02)<火星坐标>

 @param location 百度地理坐标(BD-09)
 @return 中国国测局地理坐标(GCJ-02)<火星坐标>
 */
+ (CLLocationCoordinate2D)yj_BD09ConvertToGCJ02:(CLLocationCoordinate2D)location;

/**
 百度地理坐标(BD-09) 转换成 世界标准地理坐标(WGS-84)此接口有1-2米左右的误差,需要精确定位情景慎用

 @param location 百度地理坐标(BD-09)
 @return 世界标准地理坐标(WGS-84)
 */
+ (CLLocationCoordinate2D)yj_BD09ConvertToWGS84:(CLLocationCoordinate2D)location;

附加

1、使用IOS的私有类MKLocationManager来计算。
  这个做法是有风险的,苹果不允许私有模块被直接调用。换句话说,你的软件可能会被Deny。因为是私有模块,我们需要声明这个类和我们要用到的函数,代码如下:
  

@interface MKLocationManager   
+ (id)sharedLocationManager;    // 创建并获取MKLocationManager实例
- (BOOL)chinaShiftEnabled;     // 判断IOS系统是否支持计算偏移
- (CLLocation*)_applyChinaLocationShift:(CLLocation*)arg;   // 传入原始位置,计算偏移后的位置
@end

在CLLocationManager的位置监听函数中,我们把newLocation(原始位置),转换为中国位置:

-(void) locationManager:(CLLocationManager *)manager didUpdateToLocation:(CLLocation *)newLocation fromLocation:(CLLocation *)oldLocation{

    if ([[MKLocationManager sharedLocationManager] chinaShiftEnabled]) {
        newLocation = [[MKLocationManager sharedLocationManager] _applyChinaLocationShift:newLocation];
        if (newLocation == nil) {  // 计算location好像是要联网的,软件刚启动时前几次计算会返回nil。
            return;
        }
    }
    ...
}

总结:这样(只能在IOS5以前的系统中使用),经转换后的newLocation,已经是中国的位置了。现在在映射到MKMapView上时,会显示正确的所在位置。

2、打开MKMapView的showsUserLocation功能。
  初始化MKMapView时,将属性showsUserLocation设置为YES,MKMapView会启动内置的位置监听服务,当用户位置变化时,调用delegate的回调函数:
  

- (void)mapView:(MKMapView *)mapView didUpdateUserLocation:(MKUserLocation *)userLocation{
    // 这里得到的userLocation,已经是偏移后的位置了
}

总结:这个方法不会用到IOS的私有类和函数,不会有被绝的风险。缺点可能是不能像CLLocationManager那样进行丰富的配置,至少目前我还没找到。

源码查看链接:源码地址

你可能感兴趣的:(iOS)