Geospatial 地理位置
朋友的定位,附近的人,打车距离计算?
Redis 的 Geo 在Redis3.2 版本就推出了! 这个功能可以推算地理位置的信息,两地之间的距离,方圆几里的人
这个每个城市的经纬度查询:>>http://www.jsons.cn/lngcode/
只有6个命名
Redis官方文档:https://www.redis.net.cn/order/3685.html
getadd指令
# getadd 添加地理位置
# 规则:两级无法直接添加,我们一般会下载城市数据,直接通过java程序一次性导入!
# 有效的经度从-180度到180度。
# 有效的纬度从-85.05112878度到85.05112878度。
# 当坐标位置超出上述指定范围时,该命令将会返回一个错误。
# 127.0.0.1:6379> geoadd china:city 39.90 116.40 beijin
添加城市坐标
127.0.0.1:6379> geoadd china:city 116.40 39.90 beijing
(integer) 1
127.0.0.1:6379> geoadd china:city 121.47 31.23 shanghai
(integer) 1
127.0.0.1:6379> geoadd china:city 114.05 22.52 shenzhen
(integer) 1
127.0.0.1:6379>
getpos
获得当前定位:一定是一个坐标值!
127.0.0.1:6379> geopos china:city beijing
1) 1) "116.39999896287918"
2) "39.900000091670925"
127.0.0.1:6379> geopos china:city shanghai
1) 1) "121.47000163793564"
2) "31.229999039757836"
127.0.0.1:6379> geopos china:city shenzhen
1) 1) "114.04999762773514"
2) "22.520000087950386"
127.0.0.1:6379>
GEODIST
两人之间的距离!
单位:
m 表示单位为米。
km 表示单位为千米。
mi 表示单位为英里。
ft 表示单位为英尺。
127.0.0.1:6379> geodist china:city beijing shanghai km
"1067.3788"
127.0.0.1:6379> geodist china:city beijing shenzhen km
"1945.7881"
127.0.0.1:6379>
因为添加城市经纬度的过程中,添加的时候是某个城市的区经纬度,所以存在一点误差
北京到上海直线距离:
北京到深圳直线距离
georadius
以给定的经纬度为中心, 找出某一半径内的元素
1.我附近的人? (获得所有附近的人的地址,定位!)通过半径来查询!
获得指定数量的人,200 [在朋友圈附近的人中,如果有1w的人,不可能会显示全部,所有可以指定数量]
所有数据应该都录入:china:city ,才会让结果更加直观!
127.0.0.1:6379> georadius china:city 115 30 2000 km
1) "shenzhen" >>获取115经度和30纬度为中心,半径2000km的人
2) "shanghai"
3) "beijing"
127.0.0.1:6379> georadius china:city 115 30 1000 km
1) "shenzhen"
2) "shanghai"
127.0.0.1:6379> georadius china:city 115 30 2000 km withdist
1) 1) "shenzhen" >>显示到中间距离的位置
2) "837.3386"
2) 1) "shanghai"
2) "634.1468"
3) 1) "beijing"
2) "1108.4675"
127.0.0.1:6379> georadius china:city 115 30 2000 km withcoord
1) 1) "shenzhen" >>显示他人的定位信息
2) 1) "114.04999762773514"
2) "22.520000087950386"
2) 1) "shanghai"
2) 1) "121.47000163793564"
2) "31.229999039757836"
3) 1) "beijing"
2) 1) "116.39999896287918"
2) "39.900000091670925"
127.0.0.1:6379> georadius china:city 115 30 2000 km withdist withcoord count 1 >>筛选出指定的结果!
1) 1) "shanghai"
2) "634.1468"
3) 1) "121.47000163793564"
2) "31.229999039757836"
127.0.0.1:6379> georadius china:city 115 30 2000 km withdist withcoord count 2
1) 1) "shanghai"
2) "634.1468"
3) 1) "121.47000163793564"
2) "31.229999039757836"
2) 1) "shenzhen"
2) "837.3386"
3) 1) "114.04999762773514"
2) "22.520000087950386"
GEORADIUSBYMEMBER
找出位于指定元素周围的其他元素!
127.0.0.1:6379> GEORADIUSBYMEMBER china:city beijing 2000 km
1) "shenzhen"
2) "shanghai"
3) "beijing"
127.0.0.1:6379> GEORADIUSBYMEMBER china:city beijing 1000 km
1) "beijing"
127.0.0.1:6379>
GEOHASH
返回一个或多个位置元素的 Geohash 表示
该命令将返回11个字符的Geohash字符串!
#将二维的经纬度转换为一维的字符串,如果两个字符串越接近,那么则距离越近!
127.0.0.1:6379> geohash china:city beijing shenzhen
1) "wx4fbxxfke0"
2) "ws10578st80"
总结
GEO 底层的实现原理其实就是 Zset!我们可以使用Zset命令来操作geo!
127.0.0.1:6379> zrange china:city 0 -1
1) "shenzhen"
2) "shanghai"
3) "beijing"
127.0.0.1:6379>
Hyperloglog
Redis 2.8.9 版本就更新了 Hyperloglog 数据结构!
Redis Hyperloglog 基数统计的算法!
优点:占用的内存是固定,2^64 不同的元素的技术,只需要废 12KB内存!如果要从内存角度来比较的话 Hyperloglog 首选!
网页的 UV (一个人访问一个网站多次,但是还是算作一个人!)
传统的方式, set 保存用户的id,然后就可以统计 set 中的元素数量作为标准判断 !
这个方式如果保存大量的用户id,就会比较麻烦!我们的目的是为了计数,而不是保存用户id;
0.81% 错误率! 统计UV任务,可以忽略不计的!
127.0.0.1:6379> pfadd mykey a b c d e f g h i j #创建第一组元素 mykey
(integer) 1
127.0.0.1:6379> pfcount mykey
(integer) 10
127.0.0.1:6379> pfadd mykey2 i j z x c v b n m #创建第二组元素 mykey2
(integer) 1
127.0.0.1:6379> pfcount mykey2
(integer) 9
127.0.0.1:6379> pfmerge mykey3 mykey mykey2 #合并两组 mykey mykey2 => mykey3 并集
OK
127.0.0.1:6379> pfcount mykey3 #看并集的数量!
(integer) 15
如果允许容错,那么一定可以使用 Hyperloglog !
如果不允许容错,就使用 set 或者自己的数据类型即可!
Bitmap(位存储)
统计用户信息,活跃,不活跃! 登录 、 未登录! 打卡,365打卡! 两个状态的,都可以使用
Bitmaps!
Bitmap 位图,数据结构! 都是操作二进制位来进行记录,就只有0 和 1 两个状态!
365 天 = 365 bit 1字节 = 8bit 46 个字节左右!
使用bitmap 来记录 周一到周日的打卡!
0打卡 1未打卡
127.0.0.1:6379> setbit sign 0 0 #星期1打卡
(integer) 1
127.0.0.1:6379> setbit sign 1 1 #星期2未打卡
(integer) 0
127.0.0.1:6379> setbit sign 2 0 #星期3打卡
127.0.0.1:6379> setbit sign 3 0 #星期4打卡
(integer) 0
127.0.0.1:6379> setbit sign 4 0 #星期5打卡
(integer) 0
127.0.0.1:6379> setbit sign 5 0 #星期6打卡
(integer) 0
127.0.0.1:6379> setbit sign 6 1 #星期7未打卡
(integer) 0
查看某一天是否有打卡!
127.0.0.1:6379> getbit sign 2
(integer) 0
127.0.0.1:6379> getbit sign 5
(integer) 0
127.0.0.1:6379>
统计操作,统计 未打卡的天数!
127.0.0.1:6379> bitcount sign
(integer) 2
127.0.0.1:6379>