Before:

1. 研究的需要, 在 google map 上爬取了一些的静态卫星地图图片,每张图片的像素为 256*256

2. 通过 photshop 将这些地图碎片手动拼成了地图, 地图只是覆盖了学校而已, 还是比较小的

3. 我通过手机采集到一些 GPS 的 trace, 希望在这个离线地图上画上这些 GPS 点

4. 最初, 我以为地图上的经纬度都是等距分布的, 比如一张图片的顶部纬度是20, 底部纬度为22, 那么图片中间的横线纬度是 21. 实际上, 并非如此, 中间横线的纬度应当是小于 21 的, 想了解具体的原因, 可搜索墨卡托投影

5. 好消息是, 经度是等距分布的, 比如图片左边经度是20, 右边是 22, 那么中间竖线经度是 21。 因此, 我们仅需搞定纬度

解决办法:

1. 古德曼函数 的输入是一个点距离赤道的“距离” y,输出是该点的纬度 lat。 我们的需求恰好相反, 已知 lat, 希望知道 y。 wiki 同时也提供了古德曼函数的反函数, 因此, 直接使用就好

2. 古德曼反函数有多种形式, wiki 的中文和英文版本至少提供了三种实现形式, 选一个就好, 都是等价的

3.  古德曼函数的一段 python 代码. 注意函数输入 longitude 不是 30, 60 这种值, 而是 (30/180 * math.pi) 这种类型的输入

def reverseGoodman(longitude):
return math.log((1 + math.sin(longitude)) / (1 - math.sin(longitude)))/2

输入 0 时,返回 0; 输入为  85.05112877980659, 返回的是 3.1415...(math.pi)

我本期待输入 [0, 85.05112877980659] 的输出应当是 [0, 1], 所以这个地方疑惑了挺久, 后来发现只需要在原始公式的基础上除以 math.pi 即可将值域锁定到 [0,1]

4. 墨卡托投影的假设是地球经度为 [-180, 180], 纬度为[-85.05112877980659,85.05112877980659], 投影出的地球是一个正方形, 长宽分别为 [-20037508.3427892,20037508.3427892]

5. 联合 (3, 4), 就能完成一般的计算了. 假如某点的纬度为 31度, 那么该点距离赤道的距离是

20037508.3427892/math.pi*reverseGoodman(31/180*math.pi)

  再高级一点, 假如已知两个点的纬度, 求其在经度方向上的距离差,为此, 我写了个函数

def reverseGoodman(longitude):
return math.log((1 + math.sin(longitude)) / (1 - math.sin(longitude)))/math.pi/2 def convertGeoToMeter(lat1, lat2):
lat1 = lat1 * math.pi / 180
lat2 = lat2 * math.pi / 180 worldMapHeight = 20037508.3427892
OffsetY1 = worldMapHeight * reverseGoodman(lat1)
OffsetY2 = worldMapHeight * reverseGoodman(lat2) return OffsetY2-OffsetY1

Advanced:

假设已经获得了 GPS trace, 上面的函数也已经准备好, 现在缺少的只剩静态地图的 GPS 信息了, 有了这个信息, 就能实现离线地图标注

However, google map 的经纬度并非从 0 开始计算, 不能从 zoom level + 地图碎片的索引 获得图片某一个点(如左上角)的GPS

我依然用了比较粗糙办法: 先找一个地图碎片,已知其索引和 zoom level, 找到其一个边缘的某个点, 这个点必须比较好认, 然后打开 web google map, 同样找到这个点, 右键这里是哪, 得到 GPS 信息

得到 GPS 信息后, 需要计算这种办法的误差

1. 我爬取的地图 zoom level 为 17。 根据 google map 的原理, 在 zoom level == 17 这一层, 共有 (4^17) 个静态地图碎片。 其中, 横纵方向, 各有 (2^17) 个地图碎片

2. 每张地图碎片的属性和大家熟悉的地图相同, 支持分度值, 即横纵方向上, 每一厘米表示的实际距离是相同的

3. 手动的获得一张地图碎片上下边缘的纬度, 分别为 31.027048, 31.02940。 每张地图的长度实际为 realDis, 根据经纬度之差计算的来的与真实值之差小于 0.2 米, 准确度很高, 可以使用

realDis = 20037508.3427892/(2**16)
print convertGeoToMeter(31.027048, 31.02940)*100/realDis # 99.9327328479%
print math.fabs(convertGeoToMeter(31.027048, 31.02940) - realDis) #0.205668048242

  

4. 经度的误差计算更加容易, 误差依然很小

import math

realLngDis = 360*1.0/(2**17)
observedLngDis = 121.429158-121.426381 print realLatDis
print observedLatDis
print math.fabs(realLatDis-observedLatDis)/realLatDis*100
#0.00274658203125
#0.00277699999999
#1.10748444425

  

Material:

1. 爬取的一张地图碎片

每张地图碎片含有 x, y, zoom level 三个属性

2. 地图碎片的爬取函数, python

import urllib
import os
def scheduler(a, b, c):
per = 100 * a * b / c
if per > 100:
per = 100
print "%.2f%%" %per def downloader(x, y):
for i in xrange(1,7):
for j in xrange(1,5):
url = 'https://mts2.google.com/vt/lyrs=s@139&hl=en&gl=CN&src=app&x=%i&s=&y=%i&z=17&s=' %(int(i)+ int(x), int(j) + int(y))
local = os.path.join('E:\Copy\google_map', 'x%iy%iz17.jpg' %(i+ int(x), j + int(y)) )
urllib.urlretrieve(url, local, scheduler)
#填写你需要从哪张图片开始爬取
#downloader(index_of_x, index_of_y )

 代码写的比较死, 但代码量比较小, 还是非常容易修改的

墨卡托投影, GPS 坐标转像素, GPS 坐标转距离的更多相关文章

  1. 照片提取GPS 转成百度地图坐标

    感谢: 小慧only http://www.cnblogs.com/zhaohuionly/p/3142623.html  GPS转化坐标方法 大胡子青松 http://www.cnblogs.com ...

  2. GPS坐标转换为百度地图坐标

    /** * GPS坐标转换为百度地图坐标 * 需要引入javabase64.jar 和json的一些包 * */ public class Gps2BaiDu { public static void ...

  3. GPS坐标转百度地图坐标

    百度地图提供了相关API:BMap.Convertor.translate, 但是使用上存在部分限制:1.次数限制:2.异步回调 可以用如下方法: /** * 地图位置计算工具(将GPS坐标转换成百度 ...

  4. BD09坐标(百度坐标) WGS84(GPS坐标) GCJ02(国测局坐标) 的相互转换

    BD09坐标(百度坐标) WGS84(GPS坐标) GCJ02(国测局坐标) 的相互转换 http://www.cnphp6.com/archives/24822 by root ⋅ Leave a ...

  5. WGS84、GCJ-02(火星坐标)、百度坐标,Web墨卡托坐标

    GCJ-02坐标系统(火星坐标)简介:http://blog.csdn.net/giswens/article/details/8775121(存档:http://mapbd.com/cms/2012 ...

  6. OpenGL超级宝典总结(二)2D/3D笛卡尔坐标、坐标裁剪、纹理坐标、MVP转换等概念

    如果你想把图形渲染在正确的位置上,那么坐标的设置就很重要了.在OpenGL中,与坐标相关的主要有笛卡尔坐标.坐标裁剪.纹理坐标.MVP(Model View Projection)转换. 1.笛卡尔坐 ...

  7. 验证坐标在某片坐标区域内 php 代码

    之前碰到的这样一个需求,要将公司的服务范围在地图中显示出来,并将用户每天的访问坐标进行统计看有多少用户是在所能达到的服务范围半径内. 以下是PHP代码的实现 (仅验证坐标在某片坐标区域内) <? ...

  8. OpenGL屏幕二维坐标转化成三维模型坐标

    我们把OpenGL里模型的三维坐标往二维坐标的转化称为投影,则屏幕上的二维坐标往三维坐标转化则可以称为反投影,下面我们来介绍一下反投影的方法. 主要是gluUnProject函数的使用,下面是代码: ...

  9. GPS校时器,GPS时钟装置,NTP网络时间服务器

    GPS校时器,GPS时钟装置,NTP网络时间服务器 GPS校时器,GPS时钟装置,NTP网络时间服务器 GPS校时器,GPS时钟装置,NTP网络时间服务器 GPS校时器,GPS时钟装置,NTP网络时间 ...

  10. baidu经纬度坐标与google经纬度坐标都转换

    baidu经纬度坐标与google经纬度坐标都是经过转换的.使用下面那个url可以将原始坐标或者谷歌的坐标转换成baidu的坐标http://api.map.baidu.com/ag/coord/co ...

随机推荐

  1. Solr学习之五

    一.段管理 段是一个自包含,仅可读的solr的索引的子集.一旦一个段被刷新到持久存储后,它将不会改变.当添加新文档到你的索引时候,它们被写入到新的段中.因此,在你的索引中,有很多激活的段.一次查询必须 ...

  2. jacky自问自答-数据库

    1.exists和in有什么区别? EXISTS用于检查子查询是否至少会返回一行数据,该子查询实际上并不返回任何数据,而是返回值True或False,而In子查询则是返回具体的数据值,与指定的字段比较 ...

  3. strerror() 和perror()函数

    在linux编程中,strerror()是个好东东,因为一个孤零零的errno看不出个所以然,然而strerror()返回的错误描述已经给我们解决问题提供了80%的成功率.但从安全性的角度来讲,str ...

  4. maven打包源码<转>

    Plugin: http://maven.apache.org/plugins/maven-source-plugin/ The Source Plugin has five goals: sourc ...

  5. ExtJS获取Grid的行数

    1.     grid.getSelectionModel().getCount() ;  // 获得当前选中的行数  2.     grid.getStore().getTotalCount();  ...

  6. win8.1 win10存储设备和驱动器分开显示

    win10同理如下: Windows 8.1不仅将资源管理器重命名为文件管理器,还将用户熟悉的“计算机/我的电脑”改名为“这台电脑”,同时还将原先的布局进行了重构,于是用户最终看到的是这样一个界面: ...

  7. zookeeper连接 org.apache.curator.framework.imps.CuratorFrameworkImpl Background exception was not retry-able or retry gave up [main-EventThread]

    ERROR org.apache.curator.framework.imps.CuratorFrameworkImpl Background exception was not retry-able ...

  8. Spring - BeanFactory 新旧工厂差异

    在将要被加入到spring容器中的service中,添加static静态代码块(加载类时被调用),用于判断spring中新旧bean工厂的加载性质. package com.witwicky.serv ...

  9. Hibernate-使用事务

    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletExcepti ...

  10. 2016年第七届蓝桥杯C/C++程序设计本科B组省赛

    /* 2016年第七届蓝桥杯C/C++程序设计本科B组省赛 煤球数目(结果填空) 煤球数目 有一堆煤球,堆成三角棱锥形.具体: 第一层放1个, 第二层3个(排列成三角形), 第三层6个(排列成三角形) ...