15.Redis GEO地理位置

本文详细介绍了Redis3.2版本引入的GEO数据类型,包括其基本概念(有序集合存储地理位置,经度/纬度表示),常用命令(如GEOADD、GEOPOS、GEODIST等)以及在地理围栏、附近人、配送服务等应用场景中的运用。
摘要由CSDN通过智能技术生成

目录


Redis专栏目录(点击进入…)



GEO地理位置

在Redis 3.2版本中,新增了一种数据类型:GEO,它主要用于存储地理位置信息,并对存储的信息进行操作。

Redis的GEO操作是一种基于地理位置信息进行操作的功能。它使用经度和纬度坐标来表示地理位置,支持存储地理位置信息用来实现诸如附近位置、摇一摇这类依赖于地理位置信息的功能。

GEO基本概念

GEO是一种有序集合(zset),它的每个元素都包含三个属性:
①位置名称(member)
②经度(longitude)
③纬度(latitude)

通过这些属性,可以在Redis中存储地理位置的坐标,并对它们进行一些有用的操作。
例如:
(1)计算两个位置之间的距离(geodist)
(2)获取某个位置的坐标(geopos)
(3)获取某个位置的geohash值(geohash)
(4)根据给定的经纬度坐标来获取指定范围内的地理位置集合(georadius)
(5)根据储存在位置集合里面的某个地点获取指定范围内的地理位置集合(georadiusbymember)

GEO应用场景

(1)地理围栏

通过设置地理位置的经纬度信息,可以将用户或者车辆等实体绑定在地理围栏内,当实体进出围栏时,可以触发相应的事件。

(2)附近的人

在类似于约会、社交、旅游等场景下,可以通过Redis GEO快速查询周围的人员或景点。

(3)配送服务

通过获取配送地址的经纬度信息,可以找到距离最近的配送员或仓库,并对订单进行分配。

(4)地址查找

在地图应用中,可以通过Redis GEO快速查询某个地址周围的商家或服务设施。

(5)动态信息

在滴滴、Uber等打车应用中,可以实时更新车辆的位置信息,以提供更加准确的车辆推荐和路线规划服务

GEO相关命令

(1)GEOADD:添加地理位置的坐标

将指定的地理空间位置(纬度、经度、名称)添加到指定的key中。

语法:

GEOADD key longitude latitude member [longitude latitude member ...]
参数描述
keyRedis键
longitude经度
latitude纬度
member成员

这些数据将会存储到sorted set这样的目的是为了方便使用GEORADIUS或者GEORADIUSBYMEMBER命令对数据进行半径查询等操作。

该命令以采用标准格式的参数x,y;所以经度必须在纬度之前。这些坐标的限制是可以被编入索引的,区域面积可以很接近极点但是不能索引。

GEOADD命令需要注意项:
①输入经纬度的时候要先经度后纬度
②有效的经度从-180度到180度
③有效的纬度从-85.05112878度到85.05112878度

GEOADD如何工作?
sorted set使用一种称为Geohash的技术进行填充。经度和纬度的位是交错的,以形成一个独特的52位整数。一个sorted set 的double score可以代表一个52位的整数,而不会失去精度。

这种格式允许半径查询检查的1 + 8个领域需要覆盖整个半径,并丢弃元素以外的半径。通过计算该区域的范围,通过计算所涵盖的范围,从不太重要的部分的排序集的得分,并计算得分范围为每个区域的sorted set中的查询。

使用什么样的地球模型(Earth model)?
这只是假设地球是一个球体,因为使用的距离公式是Haversine公式。这个公式仅适用于地球,而不是一个完美的球体。当在社交网站和其他大多数需要查询半径的应用中使用时,这些偏差都不算问题。但是,在最坏的情况下的偏差可能是0.5%,所以一些地理位置很关键的应用还是需要谨慎考虑。

例子:

redis> GEOADD Sicily 13.361389 38.115556 "Palermo" 15.087269 37.502669 "Catania"
(integer) 2
redis> GEODIST Sicily Palermo Catania
"166274.15156960039"
redis> GEORADIUS Sicily 15 37 100 km
1) "Catania"
redis> GEORADIUS Sicily 15 37 200 km
1) "Palermo"
2) "Catania"

(2)GEOPOS:获取地理位置的坐标

从key里返回所有给定位置元素的位置(经度和纬度)。

语法:

GEOPOS key member [member ...]
参数描述
keyRedis键
member成员

返回值:
GEOPOS 命令返回一个数组,数组中的每个项都由两个元素组成:第一个元素为给定位置元素的经度,而第二个元素则为给定位置元素的纬度。当给定的位置元素不存在时,对应的数组项为空值。

给定一个sorted set表示的空间索引,密集使用 geoadd 命令,它以获得指定成员的坐标往往是有益的。当空间索引填充通过 geoadd 的坐标转换成一个52位Geohash,所以返回的坐标可能不完全以添加元素的,但小的错误可能会出台。

因为 GEOPOS 命令接受可变数量的位置元素作为输入, 所以即使用户只给定了一个位置元素, 命令也会返回数组回复。

例子:

redis> GEOADD Sicily 13.361389 38.115556 "Palermo" 15.087269 37.502669 "Catania"
(integer) 2
redis> GEOPOS Sicily Palermo Catania NonExisting
1) 1) "13.361389338970184"
   2) "38.115556395496299"
2) 1) "15.087267458438873"
   2) "37.50266842333162"
3) (nil)

(3)GEODIST:两个给定位置之间的距离

返回两个给定位置之间的距离。如果两个位置之间的其中一个不存在, 那么命令返回空值。

语法:

GEODIST key member1 member2 [unit]

返回值:
计算出的距离会以双精度浮点数的形式被返回。 如果给定的位置元素不存在, 那么命令返回空值。

单位的参数 unit:

单位参数描述
m
km千米
mi英里
ft英尺

如果用户没有显式地指定单位参数, 那么 GEODIST 默认使用米作为单位。

GEODIST 命令在计算距离时会假设地球为完美的球形,在极限情况下,这一假设最大会造成 0.5% 的误差。

例子:

redis> GEOADD Sicily 13.361389 38.115556 "Palermo" 15.087269 37.502669 "Catania"
(integer) 2
redis> GEODIST Sicily Palermo Catania
"166274.15156960039"
redis> GEODIST Sicily Palermo Catania km
"166.27415156960038"
redis> GEODIST Sicily Palermo Catania mi
"103.31822459492736"
redis> GEODIST Sicily Foo Bar
(nil)

(4)GEOHASH:一个或多个位置元素的 Geohash 值表示

通常使用表示位置的元素使用不同的技术,使用Geohash位置52点整数编码。由于编码和解码过程中所使用的初始最小和最大坐标不同,编码的编码也不同于标准。此命令返回一个标准的Geohash

语法:

GEOHASH key member [member ...]

返回值:
一个数组,数组的每个项都是一个 geohash。命令返回的 geohash 的位置与用户给定的位置元素的位置一一对应。

Geohash字符串属性
该命令将返回11个字符的Geohash字符串,所以没有精度Geohash,损失相比,使用内部52位表示。返回的geohashes具有以下特性:
1.可以缩短从右边的字符。它将失去精度,但仍将指向同一地区。
2.可以在 geohash.org 网站使用,网址 http://geohash.org/。查询例子:http://geohash.org/sqdtr74hyu0.
3.与类似的前缀字符串是附近,但相反的是不正确的,这是可能的,用不同的前缀字符串附近

例子:

redis> GEOADD Sicily 13.361389 38.115556 "Palermo" 15.087269 37.502669 "Catania"
(integer) 2
redis> GEOHASH Sicily Palermo Catania
1) "sqc8b49rny0"
2) "sqdtr74hyu0"

(5)GEORADIUS:给定的经纬度坐标来获取指定范围内的地理位置集合

以给定的经纬度为中心,返回键包含的位置元素当中,与中心的距离不超过给定最大距离的所有位置元素。

语法:

GEORADIUS key longitude latitude radius m|km|ft|mi [WITHCOORD] [WITHDIST] [WITHHASH] [COUNT count]

返回值:
①在没有给定任何 WITH选项 的情况下,命令只会返回一个像 [“New York”,”Milan”,”Paris”] 这样的线性(linear)列表。
②在指定了 WITHCOORD、WITHDIST、WITHHASH 等选项的情况下,命令返回一个二层嵌套数组,内层的每个子数组就表示一个元素。

在返回嵌套数组时,子数组的第一个元素总是位置元素的名字。至于额外的信息,则会作为子数组的后续元素,按照以下顺序被返回:
①以浮点数格式返回的中心与位置元素之间的距离, 单位与用户指定范围时的单位一致。
②geohash 整数。
③由两个元素组成的坐标,分别为经度和纬度。.

例子:

GEORADIUS Sicily 15 37 200 km WITHCOORD WITHDIST

这样的命令返回的每个子数组都是类似以下格式的:

["Palermo","190.4424",["13.361389338970184","38.115556395496299"]]
参数描述
keyRedis键
longitude经度
latitude纬度
radius半径

单位的参数 unit:

单位参数描述
m
km千米
mi英里
ft英尺

在给定以下可选项时,命令会返回额外的信息:
(1)WITHDIST:
在返回位置元素的同时,将位置元素与中心之间的距离也一并返回。 距离的单位和用户给定的范围单位保持一致。

(2)WITHCOORD
将位置元素的经度和维度也一并返回。

(3)WITHHASH
以 52 位有符号整数的形式, 返回位置元素经过原始 geohash 编码的有序集合分值。 这个选项主要用于底层应用或者调试, 实际中的作用并不大。

命令默认返回未排序的位置元素。通过以下两个参数,用户可以指定被返回位置元素的排序方式:
(1)ASC
根据中心的位置,按照从近到远的方式返回位置元素。

(2)DESC
根据中心的位置,按照从远到近的方式返回位置元素。

在默认情况下,GEORADIUS 命令会返回所有匹配的位置元素。虽然用户可以使用 COUNT 选项去获取前 N 个匹配元素,但是因为命令在内部可能会需要对所有被匹配的元素进行处理,所以在对一个非常大的区域进行搜索时, 即使只使用 COUNT 选项去获取少量元素,命令的执行速度也可能会非常慢。 但是从另一方面来说,使用 COUNT 选项去减少需要返回的元素数量,对于减少带宽来说仍然是非常有用的。

例子:

redis> GEOADD Sicily 13.361389 38.115556 "Palermo" 15.087269 37.502669 "Catania"
(integer) 2
redis> GEORADIUS Sicily 15 37 200 km WITHDIST
1) 1) "Palermo"
   2) "190.4424"
2) 1) "Catania"
   2) "56.4413"
redis> GEORADIUS Sicily 15 37 200 km WITHCOORD
1) 1) "Palermo"
   2) 1) "13.361389338970184"
      2) "38.115556395496299"
2) 1) "Catania"
   2) 1) "15.087267458438873"
      2) "37.50266842333162"
redis> GEORADIUS Sicily 15 37 200 km WITHDIST WITHCOORD
1) 1) "Palermo"
   2) "190.4424"
   3) 1) "13.361389338970184"
      2) "38.115556395496299"
2) 1) "Catania"
   2) "56.4413"
   3) 1) "15.087267458438873"
      2) "37.50266842333162"

(6)GEORADIUSBYMEMBER:指定成员的位置被用作查询的中心获取指定范围内的地理位置集合

GEORADIUSBYMEMBER命令和GEORADIUS命令一样,都可以找出位于指定范围内的元素, 但是GEORADIUSBYMEMBER的中心点是由给定的位置元素决定的,而不是像GEORADIUS那样, 使用输入的经度和纬度来决定中心点

例子:

redis> GEOADD Sicily 13.583333 37.316667 "Agrigento"
(integer) 1
redis> GEOADD Sicily 13.361389 38.115556 "Palermo" 15.087269 37.502669 "Catania"
(integer) 2
redis> GEORADIUSBYMEMBER Sicily Agrigento 100 km
1) "Agrigento"
2) "Palermo"
  • 8
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

未禾

您的支持是我最宝贵的财富!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值