用postgreSQL做基于地理位置的app

   前言:项目中用到了postgreSQL中的earthdistance()函数功能计算地球上两点之间的距离,中文的资料太少了,我找到了一篇英文的、讲的很好的文章 ,特此翻译,希望能够帮助到以后用到earthdistance的同学。

  一、两种可用的选择

  当我们想用Postgres作为GEO函数使用时,我们通常有2中选择(据我所知):

  1.PostGIS: 为postgreSQL提供了高级GEO函数功能。我用了它一段时间,但是它对于我的需求来说太笨重了。

  2.Cube和Earthdistance: 这两个拓展为轻量级的Geo关系实体提供了简单、快速的实现方法。

  二、为什么在数据库服务器端做计算

  这是件非常明显的事。服务器存储了所有的数据,服务器拓展是用C/C++实现的,非常快。为数据表做索引也能加快计算速度。

  三、使用我的选择--Cube and EarthDistance

  作为开始,你应该先建一个数据库(我想你知道该怎么做),然后使它们能用我们的架构。 执行:

?

1
2

CREATE EXTENSION cube;
CREATE EXTENSION earthdistance;

  上面的命令创建了大约40个函数,以后我们做数据查询的时候就可以用了。

  在我们的例子中,我创建了名为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) 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) 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) float8 Returns the latitude in degrees of a point on the surface of the Earth.
longitude(earth) float8 Returns the longitude in degrees of a point on the surface of the Earth.
earth_distance(earth, earth) float8 Returns the great circle distance between two points on the surface of the Earth.
earth_box(earth, float8) 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 using earth_distance should be included in the query.

  数据库的操作可能就像下面这样:

?

1
2
3

SELECT events.id events.name, eaerthdiatance(ll_to_earth({currentuserlat}, {currentuserlng}), llto_earth(events.lat, events.lng))
as distancefromcurrentlocation FROM events
ORDER BY distancefromcurretnlocation ASC;

  这将给我们一个很nice的新闻事件列表,按他们的离我们当前位置的距离由近到远排序。第一个是离我们最近的。

  五、找到某个半径范围内的记录

  Cube和Earthdiatance拓展提供的另一个伟大的函数是earth_box(ll_to_earch($latlngcub), $radiusinmetres)。 这个函数通过简单的比较就能到找到某个半径范围内的所有记录。它是靠返回2点之间的“大圆距离”实现的。

  【译者注】大圆距离(Great circle disstance)指的是从球面的一点A出发到达球面上另一点B,所经过的最短路径的长度。一般说来,球面上任意两点A和B都可以与球心确定唯一的大圆,这个大圆被称为黎曼圆,而在大圆上连接这两点的较短的一条弧的长度就是大圆距离。如果想了解更多,请看wiki:大圆距离

  它能用于查询我们城市中所有的新闻事件:

?

1
2

SELECT events.id, events.name FROM events
WHERE earth_box({currentuserlat}, {currentuserlng}, {radiusinmetres}) @> ll_to_earth(events.lat, events.lng);

  这条查询语句仅仅会返回在radius_in_metres指定的半径范围内的记录,非常简单吧!

  六、提高查询速度

  你可能会发现上面的查询有不小的开销。以我的经验,最好对一些字段建立索引。 (下面这条语句假定你又events表, 同时events表有字段lat和lng)

?

1

CREATE INDEX ${nameofindex} on events USING gits(lltoearth(lat, lng));

  七、数据类型

  我的应用比较简单,所以我把经纬度(lat和lng)都设成了double类型。这使得我用Node.js开发起来更加快速,而不用再去自己定制针对GIST类型的解决方案。

  八、就这些!

  很神奇,对么?!?我们仅仅用常用的数据类型(double)就足以去用一些GEO函数创建基于地理位置的社交app(【译者注】知乎上的一个回答)!

  ---------------------------

  英语水平有限,如有翻译不周之处,请您指点!

时间: 2024-11-08 20:06:49

用postgreSQL做基于地理位置的app的相关文章

怎么更好地完成一个基于TOP的APP设计?

文章描述:不要让你的产品停留在实验室 抓?昂诵男枨蟆? "赢在淘宝"的应用持续飙升,在距离截止日期还剩两天的时候,已经突破250个了,这个数量还真有些出乎我们的意料.在试用这些应用的时候,发现很多让人眼前一亮的创意,但是当进一步将这个应用和市场结合的时候,就会发现出现"实用断层",也许是因为过于聚焦在"创意"本身,所以在这些应用的设计中往往忽略了用户的核心需求,或者说是对用户核心需求的关注度还不够.产品设计没有通用的理论可以参照,只是从一个终端用

基于phonegap开发app的实践

app开发告一段落,期间遇到不少问题,写篇文章记录一下. 为虾米要用phonegap 开发app,至少要考虑android和ios两个版本吧,android偶可以应付,ios表示完全木有接触过,于是时间成本.开发成本上去了.phonegap则解决了这个问题,而且对po主而言,用web开发的方式来搞app很爽啊有木有! 当然,用之前还是要调研下,基于phonegap的app有木有成功案例.大公司里腾讯的qq邮箱ios版,豆瓣的豆瓣音乐人都是基于phonegap.重点看了看豆瓣音乐人,很无耻的反编译

【翻译】基于 Create React App路由4.0的异步组件加载(Code Splitting)

基于 Create React App路由4.0的异步组件加载 本文章是一个额外的篇章,它可以在你的React app中,帮助加快初始的加载组件时间.当然这个操作不是完全必要的,但如果你好奇的话,请随意跟随这篇文章一起用Create React App和 react路由4.0的异步加载方式来帮助react.js构建大型应用. 代码分割(Code Splitting) 当我们用react.js写我们的单页应用程序时候,这个应用会变得越来越大,一个应用(或者路由页面)可能会引入大量的组件,可是有些组

3款基于地理位置的社交网络产品案例

随着智能手机和定位技术的发展,社交网络的发展也出现了变化.区别于http://www.aliyun.com/zixun/aggregation/1560.html">Facebook这种在网络上重构你现实生活社交的社交网络,基于地理位置的发现你身边的人的社交网络在不断的发展当中.通过Foursquare和Gowalla用户可以随时随地签到,通过Sonar和Banjo用户可以自动发现附近的人,通过Yobongo可以与附近的人群聊,通过Lokast可以形成一个微型社交网络.以上所有这些都是基于

觅ta:基于地理位置发布和分享实时交易信息的C2C网站

当你看到分类信息的网站铺天盖地.满屏广告的时候,也许你会感叹:这就是一个神奇的网站,而我们找不到自己所需要的.好吧,其实分类信息本身是一个很好的模式.但如果可以解决人们的问题或需求拿就更好了,现在,一些新的网站正在尝试,比如这个新的分类信息网站"多贝网".还有比如今天所介绍的"觅Ta",它是一款基于地图的关联地理位置.以需求为导向.做一个C2C的交易平台,服务.商品都可以. "觅ta"作为一个基于地理位置发布和分享实时交易信息的平台,在这些信息中

Snapchat 推出了基于地理位置的滤镜 “geofilters”

摘要: 今年早些时候Snapchat 推出了基于地理位置的滤镜 geofilters ,这款滤镜只能在被 Snapchat 收录过的特定地点使用,比如迪士尼乐园.特定的咖啡厅等等.不过这款滤镜之前只开放给Snapchat 今年早些时候Snapchat 推出了基于地理位置的滤镜 "geofilters" ,这款滤镜只能在被 Snapchat 收录过的特定地点使用,比如迪士尼乐园.特定的咖啡厅等等.不过这款滤镜之前只开放给Snapchat 开发者,今天Snapchat 在其官方博客宣布,正

国内10款基于地理位置的社交应用盘点

SOLOMO.LBS无疑是移动互联网时代最为炙手可热的两个词汇.其实这两个词真不应该是拿来说的,而是做的.就像交友一样,与其在网上沉溺于虚无缥缈的神交,不如走到线下,开拓我们的社交圈,古人云,朋友多了路好走,多交朋友可以让我们学到更多的东西.曾经有人说过这样一句话:我们与陌生人之间似乎总有一堵墙,看上去不可逾越,但常常只需要一个微笑或简单的一句Hi就可以打破,可是大部分宅男宅女们都难以跨出这一步.雷锋网此前已经陆续为大家推荐过多款社交应用,今天,小编再为大家盘点下目前国内比较流行的几款基于位置交

王秦岱推出基于地理位置拍照软件“发现”

[TechWeb消息]大头无线CEO王秦岱今日宣布,推出基于地理位置的拍照互动应用"发现".王秦岱介绍,"发现"基于SOLOMO概念,产品思路是用户有机会发现感兴趣的人在什么地点在干什么事情,并实时进行互动."发现"上传的照片必须为手机拍摄,而不能来自相册,倾向于 真实生活的实时互动.图一发现页面,用手指滑动地图到任何位置(TechWeb配图)图二动态页面,显示所关注人的动态(TechWeb配图)图三历史页面,点击进入某一用户头像可看到所有历史照

国内基于地理位置的社交应用评点

SOLOMO.LBS无疑是移动互联网时代最为炙手可热的两个词汇.其实这两个词真不应该是拿来说的,而是做的.就像交友一样,与其在网上沉溺于虚无缥缈的神交,不如走到线下,开拓我们的社交圈,古人云,朋友多了路好走,多交朋友可以让我们学到更多的东西.曾经有人说过这样一句话:我们与陌生人之间似乎总有一堵墙,看上去不可逾越,但常常只需要一个微笑或简单的一句Hi就可以打破,可是大部分宅男宅女们都难以跨出这一步.今天,为大家盘点下目前国内比较流行的几款基于位置交友的应用,希望大家能利用这些应用找到更多志同道合的