前言:项目中用到了postgresql中的earthdistance()函数功能计算地球上两点之间的距离,中文的资料太少了,我找到了一篇英文的、讲的很好的文章,特此翻译,希望能够帮助到以后用到earthdistance的同学。
一、两种可用的选择
当我们想用Postgres作为GEO函数使用时,我们通常有2中选择(据我所知):
1.PostGIS: 为postgresql提供了高级GEO函数功能。我用了它一段时间,但是它对于我的需求来说太笨重了。
2.Cube和Earthdistance: 这两个拓展为轻量级的Geo关系实体提供了简单、快速的实现方法。
二、为什么在数据库服务器端做计算
这是件非常明显的事。服务器存储了所有的数据,服务器拓展是用C/C++实现的,非常快。为数据表做索引也能加快计算速度。
三、使用我的选择--Cube and EarthDistance
作为开始,你应该先建一个数据库(我想你知道该怎么做),然后使它们能用我们的架构。 执行:
- CREATEEXTENSIONcube;
- CREATEEXTENSIONearthdistance;
在我们的例子中,我创建了名为events的表,字段有:id(serial),name(varchar 255),lat(double),lng(double)。(别忘了~~)
四、计算2个坐标之间的距离
计算2个坐标之间的距离,我们要用到earth_distance(ll_to_earth($latlngcube),ll_to_earth($latlng_cube))这个函数。 earth_distance()函数接受2组坐标值,返回值一个以米为单位的的数值。这能用于很多场景,比如根据某一位置找到离其最近的发生的新闻事件的列表。
【译者注】这里要提下几个重要的函数:(参考:http://www.postgresql.org/docs/8.3/static/earthdistance.html)
Table F-3. Cube-based earthdistance functions
Function | Returns | Description |
---|---|---|
earth() |
float8 | Returns the assumed radius of the Earth. |
sec_to_gc(float8) |
Converts the normal straight line (secant) distance between between two points on the surface of the Earth to the great circle distance between them. | |
gc_to_sec(float8) |
Converts the great circle distance between two points on the surface of the Earth to the normal straight line (secant) distance between them. | |
ll_to_earth(float8,float8) |
earth | Returns the location of a point on the surface of the Earth given its latitude (argument 1) and longitude (argument 2) in degrees. |
latitude(earth) |
Returns the latitude in degrees of a point on the surface of the Earth. | |
longitude(earth) |
Returns the longitude in degrees of a point on the surface of the Earth. | |
earth_distance(earth,earth) |
Returns the great circle distance between two points on the surface of the Earth. | |
earth_Box(earth,223); padding:0.5ex">cube |
Returns a Box suitable for an indexed search using the cube@>operator for points within a given great circle distance of a location. Some points in this Box are further than the specified great circle distance from the location,so a second check usingearth_distance should be included in the query. |
数据库的操作可能就像下面这样: