地图坐标系相互转换(WGS-84、GCJ-02、BD-09)
一、地图坐标系解释
在进行地图开发过程中,我们一般能接触到以下三种类型的地图坐标系:
WGS-84原始坐标系,一般用国际GPS纪录仪记录下来的经纬度,通过GPS定位拿到的原始经纬度,Google和高德地图定位的的经纬度(国外)都是基于WGS-84坐标系的。但是在国内是不允许直接用WGS84坐标系标注的,必须经过加密后才能使用。
GCJ-02坐标系,又名“火星坐标系”,是我国国测局独创的坐标体系,由WGS-84加密而成,在国内,必须至少使用GCJ-02坐标系,或者使用在GCJ-02加密后再进行加密的坐标系,如百度坐标系。高德和Google在国内都是使用GCJ-02坐标系,可以说,GCJ-02是国内最广泛使用的坐标系。
百度坐标系:bd-09,百度坐标系是在GCJ-02坐标系的基础上再次加密偏移后形成的坐标系,只适用于百度地图。(目前百度API提供了从其它坐标系转换为百度坐标系的API,但却没有从百度坐标系转为其他坐标系的API)。
二、坐标系转换(C#版本)
以下为WGS-84、GCJ02、BD-09的坐标系之间相互转换的方法:
/// <summary>
/// 圆周率
/// </summary>
private const double PI = 3.14159265358979324;
private const double X_PI = 3.14159265358979324 * 3000.0 / 180.0;
private const double A = 6378245.0;
private const double EE = 0.00669342162296594323;
private const double LON_BOUNDARY_MIN = 72.004;
private const double LAT_BOUNDARY_MIN = 0.8293;
private const double LON_BOUNDARY_MAX = 137.8347;
private const double LAT_BOUNDARY_MAX = 55.8271;/// <summary>
/// 是否中国境内坐标
/// </summary>
/// <param name="gpsLat"></param>
/// <param name="gpsLng"></param>
/// <returns></returns>
private static bool OutOfChina(double gpsLat, double gpsLng)
{if (gpsLng < LON_BOUNDARY_MIN || gpsLng > LON_BOUNDARY_MAX){return true;}if (gpsLat < LAT_BOUNDARY_MIN || gpsLat > LAT_BOUNDARY_MAX){return true;}return false;
}#region WGS坐标系与GCJ02坐标系互转/// <summary>
/// WGS84坐标系转GCJ02坐标系
/// </summary>
/// <param name="wgsLat">WGS坐标,纬度</param>
/// <param name="wgsLng">WGS坐标,经度</param>
/// <param name="gcjLat">GCJ02坐标,纬度</param>
/// <param name="gcjLng">GCJ02坐标,经度</param>
public static void WGS84_to_GCJ02(double wgsLat, double wgsLng, out double gcjLat, out double gcjLng)
{if (OutOfChina(wgsLat, wgsLng)){gcjLat = wgsLat;gcjLng = wgsLng;}else{double dLat = TransformLat(wgsLng - 105.0, wgsLat - 35.0);double dLon = TransformLon(wgsLng - 105.0, wgsLat - 35.0);double radLat = wgsLat / 180.0 * PI;double magic = Math.Sin(radLat);magic = 1 - EE * magic * magic;double sqrtMagic = Math.Sqrt(magic);dLat = (dLat * 180.0) / ((A * (1 - EE)) / (magic * sqrtMagic) * PI);dLon = (dLon * 180.0) / (A / sqrtMagic * Math.Cos(radLat) * PI);gcjLat = wgsLat + dLat;gcjLng = wgsLng + dLon;}
}public static void GCJ02_to_WGS84(double gcjLat, double gcjLng, out double wgsLat, out double wgsLng)
{WGS84_to_GCJ02(gcjLat, gcjLng, out wgsLat, out wgsLng);wgsLng = gcjLng * 2 - wgsLng;wgsLat = gcjLat * 2 - wgsLat;
}private static double TransformLat(double x, double y)
{double ret = -100.0 + 2.0 * x + 3.0 * y + 0.2 * y * y + 0.1 * x * y + 0.2 * Math.Sqrt(Math.Abs(x));ret += (20.0 * Math.Sin(6.0 * x * PI) + 20.0 * Math.Sin(2.0 * x * PI)) * 2.0 / 3.0;ret += (20.0 * Math.Sin(y * PI) + 40.0 * Math.Sin(y / 3.0 * PI)) * 2.0 / 3.0;ret += (160.0 * Math.Sin(y / 12.0 * PI) + 320 * Math.Sin(y * PI / 30.0)) * 2.0 / 3.0;return ret;
}private static double TransformLon(double x, double y)
{double ret = 300.0 + x + 2.0 * y + 0.1 * x * x + 0.1 * x * y + 0.1 * Math.Sqrt(Math.Abs(x));ret += (20.0 * Math.Sin(6.0 * x * PI) + 20.0 * Math.Sin(2.0 * x * PI)) * 2.0 / 3.0;ret += (20.0 * Math.Sin(x * PI) + 40.0 * Math.Sin(x / 3.0 * PI)) * 2.0 / 3.0;ret += (150.0 * Math.Sin(x / 12.0 * PI) + 300.0 * Math.Sin(x / 30.0 * PI)) * 2.0 / 3.0;return ret;
}#endregion#region 火星坐标系 (GCJ-02) 与百度坐标系 (BD-09) 的互转public static void GCJ02_to_Bd09(double gcjLat, double gcjLng, out double bdLat, out double bdLng)
{double z = Math.Sqrt(gcjLng * gcjLng + gcjLat * gcjLat) + 0.00002 * Math.Sin(gcjLat * PI);double theta = Math.Atan2(gcjLat, gcjLng) + 0.000003 * Math.Cos(gcjLng * PI);bdLng = z * Math.Cos(theta) + 0.0065;bdLat = z * Math.Sin(theta) + 0.006;
}public static void BD09_to_GCJ02(double bdLat, double bdLng, out double gcjLat, out double gcjLng)
{double x = bdLng - 0.0065, y = bdLat - 0.006;double z = Math.Sqrt(x * x + y * y) - 0.00002 * Math.Sin(y * PI);double theta = Math.Atan2(y, x) - 0.000003 * Math.Cos(x * PI);gcjLng = z * Math.Cos(theta);gcjLat = z * Math.Sin(theta);
}#endregion
三、QT/C++版本的坐标转换
创建头文件,QGCTransform.h:
#ifndef QGCTRANSFORM_H
#define QGCTRANSFORM_Hstruct gps_position_t {int32_t lat; /*< Latitude, expressed as degrees * 1E7*/int32_t lon; /*< Longitude, expressed as degrees * 1E7*/double lat_f;double lon_f;
};class QGCTransform
{
public:QGCTransform();/*WGS-84 to GCJ-02*/double PI;double a;double ee;gps_position_t gps_t;void Transform();void Transform(double lat, double lon);private:bool OutOfChina(double lat, double lon);double TransformLat(double x, double y);double TransformLon(double x, double y);
};#endif // QGCTRANSFORM_H
创建源文件,QGCTransform.cpp,转换后的GPS经纬度存储在结构体“gps_t”中,代码如下:
#include "QGCTransform.h"
#include <QtMath>QGCTransform::QGCTransform(){PI = 3.14159265358979324;a = 6378245.0;ee = 0.00669342162296594323;
}/*WGS-84 to GCJ-02*/bool QGCTransform::OutOfChina(double lat, double lon){if (lon < 72.004 || lon > 137.8347)return true;if (lat < 0.8293 || lat > 55.8271)return true;return false;
}double QGCTransform::TransformLat(double x, double y){double ret = -100.0 + 2.0 * x + 3.0 * y + 0.2 * y * y + 0.1 * x * y + 0.2 * qSqrt(qAbs(x));ret += (20.0 * qSin(6.0 * x * PI) + 20.0 * qSin(2.0 * x * PI)) * 2.0 / 3.0;ret += (20.0 * qSin(y * PI) + 40.0 * qSin(y / 3.0 * PI)) * 2.0 / 3.0;ret += (160.0 * qSin(y / 12.0 * PI) + 320 * qSin(y * PI / 30.0)) * 2.0 / 3.0;return ret;
}double QGCTransform::TransformLon(double x, double y)
{double ret = 300.0 + x + 2.0 * y + 0.1 * x * x + 0.1 * x * y + 0.1 * qSqrt(qAbs(x));ret += (20.0 * qSin(6.0 * x * PI) + 20.0 * qSin(2.0 * x * PI)) * 2.0 / 3.0;ret += (20.0 * qSin(x * PI) + 40.0 * qSin(x / 3.0 * PI)) * 2.0 / 3.0;ret += (150.0 * qSin(x / 12.0 * PI) + 300.0 * qSin(x / 30.0 * PI)) * 2.0 / 3.0;return ret;
}void QGCTransform::Transform(){double lat_s=gps_t.lat * 1E-7;double lon_s=gps_t.lon * 1E-7;Transform(lat_s, lon_s);
}void QGCTransform::Transform(double lat_s, double lon_s){gps_t.lat_f = lat_s;gps_t.lon_f = lon_s;gps_t.lat = lat_s * 1E7;gps_t.lon = lon_s * 1E7;if (OutOfChina(lat_s, lon_s )){return;}double dLat = TransformLat(lon_s - 105.0, lat_s - 35.0);double dLon = TransformLon(lon_s - 105.0, lat_s - 35.0);double radLat = lat_s / 180.0 * PI;double magic = qSin(radLat);magic = 1 - ee * magic * magic;double sqrtMagic = qSqrt(magic);dLat = (dLat * 180.0) / ((a * (1 - ee)) / (magic * sqrtMagic) * PI);dLon = (dLon * 180.0) / (a / sqrtMagic * qCos(radLat) * PI);gps_t.lat = (lat_s + dLat) * 1E7;gps_t.lon = (lon_s + dLon) * 1E7;gps_t.lat_f = lat_s + dLat;gps_t.lon_f = lon_s + dLon;
}
引用:
1、高德,百度,Google地图定位偏移以及坐标系转换
地图坐标系相互转换(WGS-84、GCJ-02、BD-09)相关推荐
- GIS中墨卡托与WGS 84的瓦片编号计算方法
GIS中墨卡托与WGS 84的瓦片编号计算方法 在GIS中计算瓦片的编号,墨卡托与WGS 84的计算方法与瓦片行列号是不一样的,为什么会有这样的差异呢?主要是因为我们的墨卡托是投影坐标系,WGS ...
- 地球坐标系(WGS - 84)转火星坐标系(GCJ)
地球坐标系(WGS - 84)转火星坐标系(GCJ) package com.tdt.convert.utils;import java.math.BigDecimal;/*** @author Mr ...
- 地图坐标系之间的转换(百度地图、GCJ、WGS84)
文章目录 文章参考 坐标系介绍 常见坐标系 WGS84坐标系(标准的GPS坐标) WGS84 Web墨卡托(平面地图) GCJ02经纬度投影(火星坐标系) GCJ02 Web 墨卡托投影 BD09 经 ...
- 【GIS - 地理信息系统】WGS 84 坐标系和 GCJ-02 坐标加密偏移 ( 大地坐标系 和 加密坐标系 | WGS 84 坐标体系简介 | GCJ-02 坐标加密偏移 )
文章目录 一.WGS 84 坐标系和 GCJ-02 坐标加密偏移 1.大地坐标系 和 加密坐标系 大地坐标系 加密坐标系 2.WGS 84 坐标体系简介 坐标加密 WGS 84 由来 经度和纬度 WG ...
- 高德地图、百度地图、腾讯地图坐标相互转换
高德地图.百度地图.腾讯地图坐标相互转换 1.WGS-84原始坐标系,一般用国际GPS纪录仪记录下来的经纬度,通过GPS定位拿到的原始经纬度,Google和高德地图定位的的经纬度(国外)都是基于WGS ...
- 百度地图和高德地图坐标系的互相转换
原文地址:http://blog.csdn.net/a13570320979/article/details/51366355 一.在进行地图开发过程中,我们一般能接触到以下三种类型的地图坐标系: 1 ...
- 坐标计算距离公式 火星坐标系_WGS84、GCJ02、BD09地图坐标系间的坐标转换及坐标距离计算...
坐标转换转载来源:http://nightfarmer.github.io/2016/12/01/GPSUtil/ 坐标系 解释 使用地图 WGS84 地球坐标系,国际上通用的坐标系.设备一般包含GP ...
- 国内地图坐标系介绍及常见地图(百度、高德、凯立德)之间的坐标系转换
一.国内地图坐标系简介 1. WGS84(地图坐标) 美国GPS使用的是WGS84的坐标系统.GPS系统获得的坐标系统,基本为标准的国际通用的WGS84坐标系统 2. GCJ-02(火星坐标) G ...
- GIS地图坐标系转换方法,快快收藏
开发地图的时最烦的就是不同的地图工具以及数据来源坐标系不同造成偏移甚至报错.以下整理了部分的地图坐标转换,一个文件直接搞定所有情况. 一.地图坐标系分类 常见的地图技术:openlayers.ces ...
最新文章
- Spoooooky CSS 选择器
- 除了Tapd,还有哪些好用的项目管理工具,适用于100+人的大中型团队?
- Media Player网页播放音频,视频,图片总汇
- linux下配置环境变量【原创】
- Linux静态IP设置
- VC环境下的静态库(lib)使用和动态库(Dll)的使用(清晰版)
- LeetCode 2141. 同时运行 N 台电脑的最长时间(二分查找)
- Windows 下的批处理脚本基础——批处理运算操作(代码可拿)
- 【转】【51CTO 网+】怎样做一款让用户来电的产品
- 20145302张薇《Java程序设计》第三周学习总结
- 400款营销策划PPT模板免费下载
- springboot 整合mybatis 报错Invalid bound statement (not found)
- 停车小程序,智能停车场小程序,智能停车源码
- 炸了~Redis bigkey导致生产事故-bigkey问题全面分析
- 学习笔记-spo0lsv病毒分析
- 优维助力国内某省级商业银行同城异地灾备自动化建设
- 最美的十大经典爱情句子{转}
- springboot中参数校验(validation)使用
- Go:Aho-Corasick 算法(附完整源码)
- Zookeeper:Zookeeper的主从选举机制
热门文章
- RLE压缩算法原理与C语言实现
- C++大数相乘(高精度乘法)
- Android-技术的下半场,Android开发者出路在哪
- c语言 mk文件详解,Android.mk/makefile 的ifeq和ifdef的区别
- NTU-Coursera机器学习:多類別分类和非线性转换
- mysql 执行概况_转mysql源码分析之SQL执行过程简介
- JavaScript数组和函数
- 高清电视HDTV概述(2)
- JimuReport积木报表 v1.5.8版本发布—免费的数据可视化报表
- 数据挖掘按技能划分,主要分为几类?