|
3 J& R$ i# d: Y8 {
l 为什么要进行坐标系转换? - Q) @" N9 p4 U4 L
因为国内对地理坐标有特殊的政策,GPS获取的原始坐标不能在互联网上对外提供,所有电子地图必须对位置做偏移处理,这就导致从底层接口得到的坐标展示在地图上会有偏移。这种偏移不是线性的,不能简单地通过加减来校正,并且不同的地图提供商采用的算法也不一样。不同的坐标系之间可能有几十到几百米的偏移,所以在地图上绘制矢量图形、做地理数据可视化时,需要修正不同坐标系之间的偏差。 2 _# y: E- c2 Y9 }$ M8 } J2 e
l 关于坐标系
/ e0 h6 w7 {/ X! G9 W 通常GPS获取的是WGS84坐标体系,包括很多国外地图厂商提供的地图数据也是WGS84坐标体系,但是在我国,由于政策原因,在地图发布和出版的时候,坐标至少需要经过国家测绘局加密,也就是对WGS84坐标进行一次非线性加偏,得到GCJ02坐标系,俗称火星坐标。 ' G+ W I( o$ |4 ]
另外,国内许多地图公司,由于各种原因,会在火星坐标的基础上进行二次非线性加偏,得到自己的坐标系统,各地图厂商使用坐标系情况如下 。
9 o; N2 \2 @! h( e3 ? 5 y+ s! [& F$ V$ S
l 坐标转换方法
7 r9 k' ]. Q9 X0 s0 e 整理了一下坐标系转换的方法,仅限单个数组转换
6 R/ o( p O. a$ C7 u4 Q3 ` 火星坐标转wgs84 3 m5 J1 @, F! {/ V1 @ |; a
// 火星坐标转wgs84! T3 }! M# M! ]) m. J' Y
function gcj02towgs84(lng, lat) {
' O8 ?1 c( R7 w& Q var x_PI = 3.14159265358979324 * 3000.0 / 180.0;9 E% L4 x: g* A8 u; Q1 r+ a
var PI = 3.1415926535897932384626;, O4 A3 c/ K, w$ D3 M+ ^8 Q
var a = 6378245.0;
: O9 u9 N8 R& m- D var ee = 0.00669342162296594323;
3 X" L/ P$ z* d6 C$ h+ J if ((lng < 72.004 || lng > 137.8347) || ((lat < 0.8293 || lat > 55.8271) || false)) {0 Z7 G, F0 G: h8 i
return [lng, lat]
" E/ y$ ]0 u+ C6 n: ~$ M }
+ K) N( N1 g& Q$ [1 Z+ | else {7 N/ f9 n, f8 T% ^! @" q5 Q
var dlat = ((lng, lat) => {3 ]3 i2 I7 H- d, U, ^" K& ~
var ret = -100.0 + 2.0 * lng + 3.0 * lat + 0.2 * lat * lat + 0.1 * lng * lat + 0.2 * Math.sqrt(Math.abs(lng));0 a( t( a9 L7 a: x! m
ret += (20.0 * Math.sin(6.0 * lng * PI) + 20.0 * Math.sin(2.0 * lng * PI)) * 2.0 / 3.0;. V6 t3 G0 j0 R$ C
ret += (20.0 * Math.sin(lat * PI) + 40.0 * Math.sin(lat / 3.0 * PI)) * 2.0 / 3.0;
# ?" z- [3 r7 Q U ret += (160.0 * Math.sin(lat / 12.0 * PI) + 320 * Math.sin(lat * PI / 30.0)) * 2.0 / 3.0;
0 k3 H" q6 i3 A. }. R return ret
9 L2 ^# x" w- M% S+ G })(lng - 105.0, lat - 35.0);9 Q2 Y$ L& K$ I8 s5 y j4 @/ ]( O
var dlng = ((lng, lat) => {* v2 H1 K# z: u* K
var ret = 300.0 + lng + 2.0 * lat + 0.1 * lng * lng + 0.1 * lng * lat + 0.1 * Math.sqrt(Math.abs(lng));
J/ L4 u4 |: C% O ret += (20.0 * Math.sin(6.0 * lng * PI) + 20.0 * Math.sin(2.0 * lng * PI)) * 2.0 / 3.0;
0 C$ E) G1 o5 d2 q$ o ret += (20.0 * Math.sin(lng * PI) + 40.0 * Math.sin(lng / 3.0 * PI)) * 2.0 / 3.0;; b- F$ g: e: I5 H% y1 s' @5 |
ret += (150.0 * Math.sin(lng / 12.0 * PI) + 300.0 * Math.sin(lng / 30.0 * PI)) * 2.0 / 3.0;
# h) \" b2 C+ v1 P7 _5 n4 x$ V return ret$ @/ O" S; @: X8 w. b! Z* s* R
})(lng - 105.0, lat - 35.0);
2 z$ X6 j5 u0 w! ^: Q, z F) H var radlat = lat / 180.0 * PI; _* g$ e' Y8 X1 B/ q. i& T6 D& r
var magic = Math.sin(radlat);7 e# _7 N: d: L) m ?0 R
magic = 1 - ee * magic * magic;
1 n3 N' @2 P; b a; p" H; q var sqrtmagic = Math.sqrt(magic);4 r1 f! O5 x" n! M7 V- Z
dlat = (dlat * 180.0) / ((a * (1 - ee)) / (magic * sqrtmagic) * PI);
# B7 ]0 D3 Z6 Q. K9 s7 k dlng = (dlng * 180.0) / (a / sqrtmagic * Math.cos(radlat) * PI);
! ~0 A& Z& N4 N7 q mglat = lat + dlat;
. H6 G" T3 I$ U; X mglng = lng + dlng;8 b; k* h! l0 u4 o4 Q/ \" V/ V
return [lng * 2 - mglng, lat * 2 - mglat]
) [' \8 [: Y' j" K& a. D' g+ Q }
3 a' ^# N& u+ e3 V } 9 o4 \: ^) a+ E/ [/ t( ]
wgs84转火星坐标
8 N( S: K6 F6 T% r // wgs84转火星坐标6 i2 [5 ^% w; ?* h' \! r4 I( [
function wgs84togcj02(lng, lat) {* v1 f2 g; ^) _+ Y; O
var x_PI = 3.14159265358979324 * 3000.0 / 180.0;# d3 P) G3 l- V( `* K/ O0 J y$ u
var PI = 3.1415926535897932384626;
- Y8 ?4 S0 L- W6 p- Y var a = 6378245.0;
3 M) ]6 n# h% C& q5 `8 ?4 C var ee = 0.00669342162296594323;
' F- }( c! b0 x3 c- f0 U0 E if ((lng < 72.004 || lng > 137.8347) || ((lat < 0.8293 || lat > 55.8271) || false)) {+ k4 b5 ~% F2 c$ x( A
return [lng, lat] q+ q* ^) }- e8 N
} else {
9 N4 p; L& M4 W8 o+ C var dlat = ((lng, lat) => {$ d |8 H' T" b% U6 `4 {4 o7 a
var ret = -100.0 + 2.0 * lng + 3.0 * lat + 0.2 * lat * lat + 0.1 * lng * lat + 0.2 * Math.sqrt(Math.abs(lng));; X- m4 k9 |0 Z" u( A
ret += (20.0 * Math.sin(6.0 * lng * PI) + 20.0 * Math.sin(2.0 * lng * PI)) * 2.0 / 3.0;$ V: b4 v" e. J$ C% r$ c: D0 ?* j1 ~
ret += (20.0 * Math.sin(lat * PI) + 40.0 * Math.sin(lat / 3.0 * PI)) * 2.0 / 3.0;
. e& s2 \& q0 w1 h' l ret += (160.0 * Math.sin(lat / 12.0 * PI) + 320 * Math.sin(lat * PI / 30.0)) * 2.0 / 3.0;/ X$ \/ y2 C5 S: z
return ret
: T A1 {0 ^: V })(lng - 105.0, lat - 35.0);1 |5 Q% ^1 o( q- r; N. z
var dlng = ((lng, lat) => {0 X: Y! W$ M& n" q3 z! @8 c
var ret = 300.0 + lng + 2.0 * lat + 0.1 * lng * lng + 0.1 * lng * lat + 0.1 * Math.sqrt(Math.abs(lng));2 [: W) |& u" ^4 p$ t" o. }+ t
ret += (20.0 * Math.sin(6.0 * lng * PI) + 20.0 * Math.sin(2.0 * lng * PI)) * 2.0 / 3.0;! ]/ r: V: g5 b6 \) l. o' V1 w
ret += (20.0 * Math.sin(lng * PI) + 40.0 * Math.sin(lng / 3.0 * PI)) * 2.0 / 3.0;
9 A% q$ ]% M0 w; ^" q4 z7 l+ _ ret += (150.0 * Math.sin(lng / 12.0 * PI) + 300.0 * Math.sin(lng / 30.0 * PI)) * 2.0 / 3.0;
$ k R8 K" l3 }5 j* L3 \+ n6 } return ret Z& X7 i8 B( Q& p
}; c7 Y( ^5 q L
)(lng - 105.0, lat - 35.0);
6 O, m4 v6 H2 U; j8 Y% i var radlat = lat / 180.0 * PI;# F3 S- E5 W0 ~- ?- I/ h/ \$ |8 E
var magic = Math.sin(radlat);
& b0 M' }6 N$ v1 x. N magic = 1 - ee * magic * magic;; [- T' k/ E. Z8 v! Z& o/ m
var sqrtmagic = Math.sqrt(magic);
) K' |/ s% `. f. ]+ I dlat = (dlat * 180.0) / ((a * (1 - ee)) / (magic * sqrtmagic) * PI);( W, t A( V% V3 m7 Y5 K
dlng = (dlng * 180.0) / (a / sqrtmagic * Math.cos(radlat) * PI);
4 G3 @ y4 f& N5 O/ t var mglat = lat + dlat;0 ^& u% p$ y3 I9 L/ O5 h' m3 l
var mglng = lng + dlng;2 a) S; Z4 @4 j* F; m* ?
return [mglng, mglat]
9 }$ I9 Q8 k% q }
, \0 `$ k4 y3 n! H* \: [; [% Y" X: R+ B } # i+ `) H2 |& a3 Q. G
百度转火星坐标 ) R5 @2 I# B, K+ F0 [
// 百度转火星坐标
4 I- T( z. T7 H2 T: |6 ~5 U const bd2xh = (bd_lng, bd_lat) => {
2 q/ e5 t& W$ W' o4 H const x_pi = Math.PI * 3000.0 / 180.0;
% q9 D- x& t9 e' [' f6 C2 o% y7 m const x = bd_lng - 0.0065;
6 d* Q6 R7 p/ f! e7 G$ s const y = bd_lat - 0.006;
: a. |# n, D2 ^. c3 s; ~1 P let z = Math.sqrt(x * x + y * y) - 0.00002 * Math.sin(y * x_pi);
# }0 R/ T- @% u* Y4 D2 M1 i* e let theta = Math.atan2(y, x) - 0.000003 * Math.cos(x * x_pi);) q5 }5 {: D- {3 q" Y- @4 G. ]& k
return [z * Math.cos(theta), z * Math.sin(theta)];
, _ | @3 T3 ] C& Y9 |3 Q }
3 S# ]4 C- U$ I3 G* s; F; n; {5 ] l 坐标转换工具
; I+ e- A& j+ X" n6 W: L- n 有的静态数据不需要动态转换也有现成的在线坐标系转换工具可以直接拿来用
, ^: B' k2 c& p7 t3 n7 _ ; G- @" V6 w1 v) a2 X
支持常见百度地图(BD09)、火星坐标系(GCJ02)、WGS84坐标系之间的相互转换 # H/ l9 m* K9 r% q6 t# {9 t# X3 X+ E
具体支持的坐标系转换如下
! G: e+ e) e0 q/ |, p, h4 Q 1)WGS-84 → GCJ-02 " V/ u! o9 Z g
2)WGS-84 → BD-09
% B* R& {! L, k1 ^ 2)GCJ-02 → WGS-84
7 m' t! P8 `# T$ w) ^7 d 3)GCJ-02 → BD-09
0 b% n! ~1 J9 v% ~* Z9 g 4)BD-09 → GCJ-02 / \3 E2 w9 z" I$ U5 r
5)BD-09 → WGS-84
/ K+ K/ B) w! A5 |( S 支持单个数组、不包含嵌套关系的对象、标准GeoJSON
! n* J$ j* a& `2 ` Q0 T 这个工具也还可以获取行政边界数据、地图数据模拟 ) ]* S( r* ^% p
7 Y% ]) {$ p4 l0 {" h% {
关注【官方微信公众号】,查看更多可视化资讯~等你呦1、EasyV数字孪生可视化2、易知微数字孪生世界 EasyV数字孪生可视化平台免费试用易知微官网,申请EasyV免费试用,个人版已上线,立即去免费体验!
. @* U2 k# i/ m9 C5 M. | 免费试用期间:运营、产品、设计、技术人员专业一对一拉群指导,解答疑惑。 % w2 v/ S; }: l6 [. H4 ?0 J
! V5 t2 e9 H! o4 X$ d+ k5 w% N" o- [
$ R2 }7 ?% @' J8 r+ F" d* @6 ]: N$ |. h
3 y0 c) `6 a$ b7 f, n1 }) M& k; b8 c G" a% ^! | ?
|