现在的位置: 首页 > 综合 > 正文

CLLocationManager和中国地图偏移

2018年01月08日 ⁄ 综合 ⁄ 共 1513字 ⁄ 字号 评论关闭

CLLocationManager类的作用是监听GPS的位置消息,当用户坐标发生变化时,会调用下面的方法进行通知:
-(void) locationManager:(CLLocationManager *)manager didUpdateToLocation:(CLLocation *)newLocation fromLocation:(CLLocation
*)oldLocation;

问题来了,当我们把newLocation中的坐标映射到MKMapView控件上时,会发现这个点跟本不是我们所在的位置,而是离我们百米左右的某个地方。
至于原因,。
那么,我们需要将原始的(真实的)位置坐标,转换为有中国特色的坐标。
对于这个问题,本文总结了两个处理办法:
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;
        }
    }
    ...
}
  这样,经转换后的newLocation,已经是中国的位置了。现在在映射到MKMapView上时,会显示正确的所在位置。

2. 打开MKMapView的showsUserLocation功能。
  初始化MKMapView时,将属性showsUserLocation设置为YES,MKMapView会启动内置的位置监听服务,当用户位置变化时,调用delegate的回调函数:
- (void)mapView:(MKMapView *)mapView didUpdateUserLocation:(MKUserLocation *)userLocation
{
    // 这里得到的userLocation,已经是偏移后的位置了
}
  这个方法不会用到IOS的私有类和函数,不会有被绝的风险。缺点可能是不能像CLLocationManager那样进行丰富的配置,至少目前我还没找到。 

抱歉!评论已关闭.