You've already forked geo-converter
mirror of
https://github.com/qqAys/geo-converter.git
synced 2025-08-01 15:04:44 +08:00
Create geo-converter.py
This commit is contained in:
93
geo-converter.py
Normal file
93
geo-converter.py
Normal file
@@ -0,0 +1,93 @@
|
||||
import math
|
||||
|
||||
def gcj02_to_wgs84(gcj_lon, gcj_lat):
|
||||
"""
|
||||
GCJ-02坐标转换为WGS-84坐标(近似反向转换)
|
||||
"""
|
||||
# 迭代逼近法
|
||||
wgs_lon, wgs_lat = gcj_lon, gcj_lat
|
||||
for i in range(30):
|
||||
tmp_lon, tmp_lat = wgs84_to_gcj02(wgs_lon, wgs_lat)
|
||||
wgs_lon = wgs_lon - (tmp_lon - gcj_lon)
|
||||
wgs_lat = wgs_lat - (tmp_lat - gcj_lat)
|
||||
return wgs_lon, wgs_lat
|
||||
|
||||
def wgs84_to_gcj02(wgs_lon, wgs_lat):
|
||||
"""
|
||||
WGS-84坐标转换为GCJ-02坐标(火星坐标系)
|
||||
|
||||
Args:
|
||||
wgs_lon (float): WGS-84经度
|
||||
wgs_lat (float): WGS-84纬度
|
||||
|
||||
Returns:
|
||||
tuple: (gcj_lon, gcj_lat) GCJ-02坐标
|
||||
"""
|
||||
|
||||
# 判断是否在中国范围内
|
||||
def out_of_china(lon, lat):
|
||||
if lon < 72.004 or lon > 137.8347:
|
||||
return True
|
||||
if lat < 0.8293 or lat > 55.8271:
|
||||
return True
|
||||
return False
|
||||
|
||||
# 如果不在中国范围内,直接返回原坐标
|
||||
if out_of_china(wgs_lon, wgs_lat):
|
||||
return wgs_lon, wgs_lat
|
||||
|
||||
# 转换参数
|
||||
a = 6378245.0 # 长半轴
|
||||
ee = 0.00669342162296594323 # 扁率
|
||||
|
||||
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
|
||||
|
||||
# 计算偏移量
|
||||
d_lat = transform_lat(wgs_lon - 105.0, wgs_lat - 35.0)
|
||||
d_lon = transform_lon(wgs_lon - 105.0, wgs_lat - 35.0)
|
||||
|
||||
# 纬度弧度
|
||||
rad_lat = wgs_lat / 180.0 * math.pi
|
||||
magic = math.sin(rad_lat)
|
||||
magic = 1 - ee * magic * magic
|
||||
sqrt_magic = math.sqrt(magic)
|
||||
|
||||
# 调整偏移量
|
||||
d_lat = (d_lat * 180.0) / ((a * (1 - ee)) / (magic * sqrt_magic) * math.pi)
|
||||
d_lon = (d_lon * 180.0) / (a / sqrt_magic * math.cos(rad_lat) * math.pi)
|
||||
|
||||
# 计算最终坐标
|
||||
gcj_lat = wgs_lat + d_lat
|
||||
gcj_lon = wgs_lon + d_lon
|
||||
|
||||
return gcj_lon, gcj_lat
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
wgs_lon = 116.397428 # 北京天安门经度
|
||||
wgs_lat = 39.90923 # 北京天安门纬度
|
||||
|
||||
print(f"WGS-84坐标: ({wgs_lon}, {wgs_lat})")
|
||||
|
||||
gcj_lon, gcj_lat = wgs84_to_gcj02(wgs_lon, wgs_lat)
|
||||
|
||||
print(f"GCJ-02坐标: ({gcj_lon}, {gcj_lat})")
|
||||
print(f"偏移量: ({gcj_lon - wgs_lon}, {gcj_lat - wgs_lat})")
|
||||
|
||||
wgs_lon, wgs_lat = gcj02_to_wgs84(gcj_lon, gcj_lat)
|
||||
|
||||
print(f"WGS-84坐标(转换): ({wgs_lon}, {wgs_lat})")
|
Reference in New Issue
Block a user