Contents
  1. 1. 火星坐标
  2. 2. 地球坐标 –> 火星坐标
    1. 2.1. GetMarsCoor.h
    2. 2.2. GetMarsCoor.m

不知道大家有没有遇到过这样的情况,在APP上开启定位,会发现自己所在的位置和地图上会有一定的偏差,前2年的时候这个问题特别严重,记得快的打车(那会儿还没有和滴滴合并)的坐标也会出现这种问题。我之所以记录这个是发现不少记录运动轨迹的应用(例如乐心),包括一大票水土不服的国外优秀运动APP不少还有这个问题。

为什么会出现这样的情况呢?

火星坐标

我们先来认识一个只有在我大天朝才会出现的词:火星坐标
什么叫做火星坐标?就是 不是地球上的真实坐标就对了。 废话。。。原因是我大天朝上面的所有东西都是国家机密!嘘,不要问我为什么知道。所以,我们的地图要经过偏移处理,你在经过我天朝测绘局处理的地图上看到的建筑物,和它实际所在的位置,其实是有出入的。
也不知道这群人搞出这么自欺欺人的东西目的何在,Google Earth上面看的清清楚楚的,不太明白有什么用处。。。反正就是给码农添堵就是了。

与之对应的,当然就是地球坐标了,就是除了在天朝之外的地球上,大家都统一使用的一套经纬度坐标。

举个栗子:
假如你开发了一个APP程序,那你获取当前设备的GPS信息,那获取到的这个坐标就是地球坐标,好了,如果你把这个坐标设置在经过我大天朝认可的移动设备地图上后,就会出现偏差。因为在经过偏移后的地图上面,只有把你获取到的坐标经过处理,换算成火星坐标,才能在地图上显示正确的位置。

那说到这里,我们都很明白了,如果自己开发一款程序需要自己获取经纬度坐标,然后在地图上打点,就需要经过 地球坐标---> 火星坐标 之间的转换,然后在地图上把,火星坐标打上去,才让用户觉得你的这个坐标是正确的。

地球坐标 –> 火星坐标

直接上代码:

GetMarsCoor.h

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
//
// GetMarsCoor.h
//
// Created by KaKa on 15/3/9.
// Copyright (c) 2015年 KAKA. All rights reserved.
//
// 将地球坐标转换成火星坐标

#import <Foundation/Foundation.h>
#import <MapKit/MapKit.h>


@interface GetMarsCoor : NSObject
+ (CLLocation *)transformToMars:(CLLocation *)location;
+ (BOOL)outOfChina:(CLLocation *)location;
+ (double)transformLatWithX:(double)x y:(double)y;
+ (double)transformLonWithX:(double)x y:(double)y;
@end

GetMarsCoor.m

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
//
// GetMarsCoor.m
//
// Created by KaKa on 15/3/9.
// Copyright (c) 2015年 KAKA. All rights reserved.
//

#import "GetMarsCoor.h"

@implementation GetMarsCoor

+ (CLLocation *)transformToMars:(CLLocation *)location {
const double a = 6378245.0;
const double ee = 0.00669342162296594323;
// 判断是否在中国大陆之外
if ([[self class] outOfChina:location]) {
return location;
}
double dLat = [[self class] transformLatWithX:location.coordinate.longitude - 105.0 y:location.coordinate.latitude - 35.0];
double dLon = [[self class] transformLonWithX:location.coordinate.longitude - 105.0 y:location.coordinate.latitude - 35.0];
double radLat = location.coordinate.latitude / 180.0 * M_PI;
double magic = sin(radLat);
magic = 1 - ee * magic * magic;
double sqrtMagic = sqrt(magic);
dLat = (dLat * 180.0) / ((a * (1 - ee)) / (magic * sqrtMagic) * M_PI);
dLon = (dLon * 180.0) / (a / sqrtMagic * cos(radLat) * M_PI);
return [[CLLocation alloc] initWithLatitude:location.coordinate.latitude + dLat longitude:location.coordinate.longitude + dLon];
}

+ (BOOL)outOfChina:(CLLocation *)location {
if (location.coordinate.longitude < 72.004 || location.coordinate.longitude > 137.8347) {
return YES;
}
if (location.coordinate.latitude < 0.8293 || location.coordinate.latitude > 55.8271) {
return YES;
}
return NO;
}

+ (double)transformLatWithX:(double)x y:(double)y {
double ret = -100.0 + 2.0 * x + 3.0 * y + 0.2 * y * y + 0.1 * x * y + 0.2 * sqrt(abs(x));
ret += (20.0 * sin(6.0 * x * M_PI) + 20.0 * sin(2.0 * x * M_PI)) * 2.0 / 3.0;
ret += (20.0 * sin(y * M_PI) + 40.0 * sin(y / 3.0 * M_PI)) * 2.0 / 3.0;
ret += (160.0 * sin(y / 12.0 * M_PI) + 320 * sin(y * M_PI / 30.0)) * 2.0 / 3.0;
return ret;
}

+ (double)transformLonWithX:(double)x y:(double)y {
double ret = 300.0 + x + 2.0 * y + 0.1 * x * x + 0.1 * x * y + 0.1 * sqrt(abs(x));
ret += (20.0 * sin(6.0 * x * M_PI) + 20.0 * sin(2.0 * x * M_PI)) * 2.0 / 3.0;
ret += (20.0 * sin(x * M_PI) + 40.0 * sin(x / 3.0 * M_PI)) * 2.0 / 3.0;
ret += (150.0 * sin(x / 12.0 * M_PI) + 300.0 * sin(x / 30.0 * M_PI)) * 2.0 / 3.0;
return ret;
}

@end

亲测这套代码在iOS上面是可以完美转换地球坐标到火星坐标。
因为年代久远还是用objective-c写的,不过在swift中使用也没有问题吧。
我刚想起来博客地址迁移,没有把在Swift中使用objective-c编写的库的方法迁移过来。回头把那篇文章迁移过来,然后再更新下链接吧~

Contents
  1. 1. 火星坐标
  2. 2. 地球坐标 –> 火星坐标
    1. 2.1. GetMarsCoor.h
    2. 2.2. GetMarsCoor.m