最近在做一个项目,有一个功能想要实现类似于查询附近的人的功能。由于项目的原因数据库只能使用 postgresql,空间查询就使用了 postgis 来实现。

具体业务像这样:业务需要返回附近距自己 1000 米的人的一个列表,返回列表中要带上距离,就是说某人距你多少米?

使用定位方式是什么?

我们使用 gps 定位发现定位精度,时间,误差和抖动都需要发大量的时间去处理。

于是采用了百度的定位 SDK,百度的上面几处都处理不错。

有一个小问题,从百度拿到的坐标是 经度和纬度,这个坐标系应该是地理坐标,由于也是使用 gps ,其基础坐标应该是 wgs84。所以拿到的值应该是:wgs84 + 国家偏移 + 百度偏移 = 百度坐标

后面两项是没有太好的办法解决,我们拿百度坐标与 google 的大致对比了一下,误差 50 米以内,业务可以接受。

而且这个后面两项是系统偏移,不是随机的,这样的话,计算距离这种相对计算,应该可以消除部分系统偏移的影响。

再说说postgis 计算距离

postgis 是一个地理空间查询引擎,计算地理距离这很方便。

  1. 开始我们在表中建一列用于存地理类型的 geometry
SELECT AddGeometryColumn ('public','chest','position',4326,'POINT',2, false);
  1. 然后保存的时候,将 经纬度和 经纬度变成 geometry 进行保存,也就是三个字段,其中保存 geometry 是用下面的sql
ST_GeomFromText('POINT(${lng} ${lat})', 4326)

sql 中的 4326 是指的wsg84 的系统。

  1. 然后对保存后的值进行距离查询,使用 ST_distance 进行,后面有二个坑等着我们:

第一个:geometry 的问题

postgisgeomery 使用 ST_distance 计算出来的单位竟然是弧度,不是 ,找了一段时间问题,发现是没有使用投影坐标引起的。

于是在计算的时候,先将 变成 投影坐标,再来计算。

ST_Transform(geo, 2346)

这里的 2346中国西安 80 高斯克里格投影的编号,是分带的,使用的中央经线在南京。

加上这个,我们能正确计算出来距离,也是

然后,一段时间后,把应用发到别的城市,也就是苏州,出现了问题

第二个: 投影带号的问题

原因是苏州不在 2346 中的带号里面,计算就出错了!

这样就有问题了,全国这么多的带号,不可能写在程序里,动态计算啊?再说万一出国了,没有中国的投影怎么办?

回头再去找方案,这个计算距离应该要支持最低就是全国,最好是全世界。

还真找到了,就是 geography

解释一下: geographygeometry 都是 postgis 中的数据类型,翻绎过来:地理图形和几何图形,用 postgis 的说法就是, geography 就是使用 wgs84 坐标系 的图形。

postgis 可以直接使用 geography 进行计算,支持全球。于是这个问题,终于有解决方案了,就是把我们的 geometry 转成 geography ,然后进行距离计算。

下面是我们使用 sql 语句

   select
(ST_distance(position::geography, ST_GeomFromText('POINT(121 32)', 4326)::geography) as distance
from chest
where ST_dwithin(position::geography, ST_GeomFromText('POINT(121 32})', 4326)::geography, 1000)
orderBy distance desc

其中:position 是我存在数据库中的字段,使用的是 geometry 的类型, (121,32 )是我输入的中心点。

::geography 就是postgis 中的 转换类型语法,把 geometry 转成 geography,然后计算,这样全球都能使用。

总结

这些问题的解决,算是把以前的知识在复习了一下。 在项目中,我们也可以直接把坐标存成 geography,使用如下的 sql

ST_GeographyFromText('SRID=4326;POINT(-110 30)')

数据就不用转来转去了。

只是项目已有数据,不能这样做了。

参考文档

PostGIS-04.md

Geography

postgis 中的距离计算的更多相关文章

  1. [转] MachingLearning中的距离相似性计算以及python实现

    参考:https://blog.csdn.net/gamer_gyt/article/details/75165842#t16  https://blog.csdn.net/ymlgrss/artic ...

  2. 你真的会用PostGIS中的buffer缓冲吗?

    buffer - 图形缓冲区分析,GIS中最基本的空间分析之一. 实现buffer的工具有很多种,例如前端的truf.js.服务端的ArcGISserver.桌面端的ArcMap.数据库端的PosrG ...

  3. 管道设计CAD系统中重量重心计算

    管道设计CAD系统中重量重心计算 eryar@163.com Abstract. 管道设计CAD系统中都有涉及到重量重心计算的功能,这个功能得到的重心数据主要用于托盘式造船时方便根据重心设置吊装配件. ...

  4. Scipy教程 - 距离计算库scipy.spatial.distance

    http://blog.csdn.net/pipisorry/article/details/48814183 在scipy.spatial中最重要的模块应该就是距离计算模块distance了. fr ...

  5. Touch事件详解及区别,触屏滑动距离计算

    移动端有四个关于触摸的事件,分别是touchstart.touchmove.touchend.touchcancel(比较少用), 它们的触发顺序是touchstart-->touchmove- ...

  6. 相似度与距离计算python代码实现

    #定义几种距离计算函数 #更高效的方式为把得分向量化之后使用scipy中定义的distance方法 from math import sqrt def euclidean_dis(rating1, r ...

  7. 通过经纬度坐标计算距离的方法(经纬度距离计算)ZZ

    通过经纬度坐标计算距离的方法(经纬度距离计算) 最近在网上搜索“通过经纬度坐标计算距离的方法”,发现网上大部分都是如下的代码: #define PI 3.14159265 static double ...

  8. V-rep学习笔记:碰撞检测与距离计算

    V-REP可以在几何组件中快速判断各种干扰与碰撞,以及计算多个组件间的最小距离. 碰撞检测 V-REP可以检测两个碰撞体实体(Collidable objects are objects that c ...

  9. 基于js利用经纬度进行两地的距离计算(转)

    转自:http://www.storyday.com/html/y2009/2212_according-to-latitude-and-longitude-distance-calculation- ...

随机推荐

  1. 【教程】发布NAServer到ArcGIS Server 10.4上[超详细]

    前阵子对ArcGIS API For JavaScript的网络分析有兴趣,但是不知道其数据是如何获取的. 查阅API知道,AJS的网络分析只有三个功能:最短路径(RouteTask).最近设施点(C ...

  2. 计算机的启动和Linux的启动

    计算机的启动和Linux的启动 一 计算机的启动   计算机的启动过程分为四个阶段,分别是:BIOS.MBR.启动管理程序.加载操作系统内核.操作系统启动.   1.1 BIOS 计算机加电后,第一件 ...

  3. C#基础知识-基本的流程控制语句(三)

    所谓的流程控制就是在程序运行中控制程序的走向,可以通过各种的条件判断执行代码的顺序,有if... if...else.. else...if |switch case...|while... Do.. ...

  4. 在SCIKIT中做PCA 逆运算 -- 新旧特征转换

    PCA(Principal Component Analysis)是一种常用的数据分析方法.PCA通过线性变换将原始数据变换为一组各维度线性无关的表示,可用于提取数据的主要特征分量,常用于高维数据的降 ...

  5. IOS开发常用的基础方法

    .//退出键盘 [self.view endEditing:YES]; 隐藏手机上方的状态栏 -(BOOL)prefersStatusBarHidden{ return YES; } //获取当前控制 ...

  6. MapControl Application 添加自定义的工具条

    现在想用二次开发做一些东西,然后需要自定义的工具条,但是如何向MapControl Application 添加自定义的工具条呢,经过多次试验后,终于找到了相应的方法(左图是添加自定义的工具条之前,右 ...

  7. fir.im 持续集成技术实践

    互联网时代,人人都在追求产品的快速响应.快速迭代和快速验证.不论是创业团队还是大中型企业,都在探索属于自己的敏捷开发.持续交付之道.fir.im 团队也在全面实施敏捷,并推出新持续集成服务 - flo ...

  8. Gephi安装

    Gephi for mac https://gephi.org/users/download/ 在官网上下载gephi-0.9.1-macos.dmg双击拖到Application里面就好了,注意有的 ...

  9. 模拟实现简化版List迭代器&嵌入List

    1.迭代器(iterators)概念(1)迭代器是一种抽象的设计概念,其定义为:提供一种方法,使他能够按顺序遍历某个聚合体(容器)所包含的所有元素,但又不需要暴露该容器的内部表现方式. (2)迭代器是 ...

  10. zabbix监控Elasticsearch集群

    本节以 zabbix 为例,介绍如何使用监控系统完成 Elasticsearch 的监控报警. github 上有好几个版本的 ESZabbix 仓库,都源自 Elastic 公司员工 unterge ...