93 lines
3.0 KiB
Python
93 lines
3.0 KiB
Python
#!/usr/bin/env python
|
|
# -*-coding:utf-8 -*-
|
|
|
|
'''
|
|
# @Author :幸运锦鲤
|
|
# @Time : 2025-03-07 00:14:56
|
|
# @version : python3
|
|
# @Update time :
|
|
# @Description : 百度地图坐标转换示例代码
|
|
'''
|
|
|
|
import math
|
|
|
|
def bd09_to_gcj02(bd_lon, bd_lat):
|
|
"""
|
|
将 BD-09 坐标系转换为 GCJ-02 坐标系
|
|
:param bd_lon: BD-09 经度
|
|
:param bd_lat: BD-09 纬度
|
|
:return: GCJ-02 坐标 (lon, lat)
|
|
"""
|
|
x = bd_lon - 0.0065
|
|
y = bd_lat - 0.006
|
|
z = math.sqrt(x * x + y * y) - 0.00002 * math.sin(y * math.pi)
|
|
theta = math.atan2(y, x) - 0.000003 * math.cos(x * math.pi)
|
|
gcj_lon = z * math.cos(theta)
|
|
gcj_lat = z * math.sin(theta)
|
|
return gcj_lon, gcj_lat
|
|
|
|
def gcj02_to_wgs84(gcj_lon, gcj_lat):
|
|
"""
|
|
将 GCJ-02 坐标系转换为 WGS-84 坐标系
|
|
:param gcj_lon: GCJ-02 经度
|
|
:param gcj_lat: GCJ-02 纬度
|
|
:return: WGS-84 坐标 (lon, lat)
|
|
"""
|
|
a = 6378245.0 # 克拉索夫斯基椭球参数长半轴
|
|
ee = 0.00669342162296594323 # 克拉索夫斯基椭球参数第一偏心率平方
|
|
PI = math.pi
|
|
|
|
# 转换公式
|
|
dLat = transform_lat(gcj_lon - 105.0, gcj_lat - 35.0)
|
|
dLon = transform_lon(gcj_lon - 105.0, gcj_lat - 35.0)
|
|
radLat = gcj_lat / 180.0 * PI
|
|
magic = math.sin(radLat)
|
|
magic = 1 - ee * magic * magic
|
|
sqrtMagic = math.sqrt(magic)
|
|
dLat = (dLat * 180.0) / ((a * (1 - ee)) / (magic * sqrtMagic) * PI)
|
|
dLon = (dLon * 180.0) / (a / sqrtMagic * math.cos(radLat) * PI)
|
|
wgs_lon = gcj_lon - dLon
|
|
wgs_lat = gcj_lat - dLat
|
|
return wgs_lon, wgs_lat
|
|
|
|
def transform_lat(x, y):
|
|
"""
|
|
纬度转换公式
|
|
"""
|
|
ret = -100.0 + 2.0 * x + 3.0 * y + 0.2 * y * y + 0.1 * x * y + 0.2 * math.sqrt(abs(x))
|
|
ret += (20.0 * math.sin(6.0 * x * math.pi) + 20.0 * math.sin(2.0 * x * math.pi)) * 2.0 / 3.0
|
|
ret += (20.0 * math.sin(y * math.pi) + 40.0 * math.sin(y / 3.0 * math.pi)) * 2.0 / 3.0
|
|
ret += (160.0 * math.sin(y / 12.0 * math.pi) + 320 * math.sin(y * math.pi / 30.0)) * 2.0 / 3.0
|
|
return ret
|
|
|
|
def transform_lon(x, y):
|
|
"""
|
|
经度转换公式
|
|
"""
|
|
ret = 300.0 + x + 2.0 * y + 0.1 * x * x + 0.1 * x * y + 0.1 * math.sqrt(abs(x))
|
|
ret += (20.0 * math.sin(6.0 * x * math.pi) + 20.0 * math.sin(2.0 * x * math.pi)) * 2.0 / 3.0
|
|
ret += (20.0 * math.sin(x * math.pi) + 40.0 * math.sin(x / 3.0 * math.pi)) * 2.0 / 3.0
|
|
ret += (150.0 * math.sin(x / 12.0 * math.pi) + 300.0 * math.sin(x / 30.0 * math.pi)) * 2.0 / 3.0
|
|
return ret
|
|
|
|
def bd09_to_wgs84(bd_lon, bd_lat):
|
|
"""
|
|
将 BD-09 坐标系转换为 WGS-84 坐标系
|
|
:param bd_lon: BD-09 经度
|
|
:param bd_lat: BD-09 纬度
|
|
:return: WGS-84 坐标 (lon, lat)
|
|
"""
|
|
# 先将 BD-09 转换为 GCJ-02
|
|
gcj_lon, gcj_lat = bd09_to_gcj02(bd_lon, bd_lat)
|
|
# 再将 GCJ-02 转换为 WGS-84
|
|
wgs_lon, wgs_lat = gcj02_to_wgs84(gcj_lon, gcj_lat)
|
|
return wgs_lon, wgs_lat
|
|
|
|
# 示例
|
|
bd_lon, bd_lat = 116.510484,39.896936 # 百度地图坐标
|
|
wgs_lon, wgs_lat = bd09_to_wgs84(bd_lon, bd_lat)
|
|
print(f"WGS-84 坐标: ({wgs_lon}, {wgs_lat})")
|
|
|
|
|
|
|