时间:2021-05-19
前言
LBS(基于位置的服务) 服务是现在移动互联网中比较常用的功能。例如外卖服务中常用的我附近的店铺的功能,通常是以用户当前的位置坐标为基础,查询一定距离范围类的店铺,按照距离远近进行倒序排序。
自从 redis 4 版本发布后, lbs 相关命令正式内置在 redis 的发行版中。要实现上述的功能,主要用到 redis geo 相关的两个命令
GEOADD 和 GEORADIOUS
命令描述
GEOADD
GEOADD key longitude latitude member [longitude latitude member ...]
这个命令将指定的地理空间位置(纬度、经度、名称)添加到指定的 key 中。
有效的经度从-180度到180度。
有效的纬度从-85.05112878度到85.05112878度。
当坐标位置超出上述指定范围时,该命令将会返回一个错误。
该命令可以一次添加多个地理位置点
GEORADIOUS
GEORADIUS key longitude latitude radius m|km|ft|mi [WITHCOORD] [WITHDIST] [WITHHASH] [COUNT count]
这个命令以给定的经纬度为中心, 返回键包含的位置元素当中, 与中心的距离不超过给定最大距离的所有位置元素。
范围可以使用以下其中一个单位:
在给定以下可选项时, 命令会返回额外的信息:
接口定义
package com.x9710.common.redis;import com.x9710.common.redis.domain.GeoCoordinate;import com.x9710.common.redis.domain.Postion;import java.util.List;public interface LBSService {/*** 存储一个位置** @param postion 增加的位置对象* @throws Exception*/boolean addPostion(Postion postion);/*** 查询以指定的坐标为中心,指定的距离为半径的范围类的所有位置点** @param center 中心点位置* @param distinct 最远距离,单位米* @param asc 是否倒序排序* @return 有效的位置*/List<Postion> radious(String type, GeoCoordinate center, Long distinct, Boolean asc);}实现的接口
测试用例
package com.x9710.common.redis.test;import com.x9710.common.redis.RedisConnection;import com.x9710.common.redis.domain.GeoCoordinate;import com.x9710.common.redis.domain.Postion;import com.x9710.common.redis.impl.CacheServiceRedisImpl;import com.x9710.common.redis.impl.LBSServiceRedisImpl;import org.junit.Assert;import org.junit.Before;import org.junit.Test;import java.util.List;/*** LBS服务测试类** @author 杨高超* @since 2017-12-28*/public class RedisLBSTest {private CacheServiceRedisImpl cacheService;private LBSServiceRedisImpl lbsServiceRedis;private String type = "SHOP";private GeoCoordinate center;@Beforepublic void before() {RedisConnection redisConnection = RedisConnectionUtil.create();lbsServiceRedis = new LBSServiceRedisImpl();lbsServiceRedis.setDbIndex(15);lbsServiceRedis.setRedisConnection(redisConnection);Postion postion = new Postion("2017122801", type, 91.118970, 29.654210);lbsServiceRedis.addPostion(postion);postion = new Postion("2017122802", type, 116.373472, 39.972528);lbsServiceRedis.addPostion(postion);postion = new Postion("2017122803", type, 116.344820, 39.948420);lbsServiceRedis.addPostion(postion);postion = new Postion("2017122804", type, 116.637920, 39.905460);lbsServiceRedis.addPostion(postion);postion = new Postion("2017122805", type, 118.514590, 37.448150);lbsServiceRedis.addPostion(postion);postion = new Postion("2017122806", type, 116.374766, 40.109508);lbsServiceRedis.addPostion(postion);center = new GeoCoordinate();center.setLongitude(116.373472);center.setLatitude(39.972528);}@Testpublic void test10KMRadious() {List<Postion> postions = lbsServiceRedis.radious(type, center, 1000 * 10L, true);Assert.assertTrue(postions.size() == 2 && exist(postions, "2017122802") && exist(postions, "2017122803"));}@Testpublic void test50KMRadious() {List<Postion> postions = lbsServiceRedis.radious(type, center, 1000 * 50L, true);Assert.assertTrue(postions.size() == 4&& exist(postions, "2017122802")&& exist(postions, "2017122803")&& exist(postions, "2017122806")&& exist(postions, "2017122804"));}private boolean exist(List<Postion> postions, String key) {if (postions != null) {for (Postion postion : postions) {if (postion.getId().equals(key)) {return true;}}}return false;}@Beforepublic void after() {RedisConnection redisConnection = RedisConnectionUtil.create();cacheService = new CacheServiceRedisImpl();cacheService.setDbIndex(15);cacheService.setRedisConnection(redisConnection);cacheService.delObject(type);}}测试结果
LBS 服务测试结果
后记
这样,我们通过 redis 就能简单实现一个我附近的小店的功能的 LBS服务。
代码同步发布在 GitHub 仓库中
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
这段时间一直有人问如何在Redis中缓存Java中的List集合数据,其实很简单,常用的方式有两种:1.利用序列化,把对象序列化成二进制格式,Redis提供了相
本文介绍了Java利用Redis实现消息队列的示例代码,分享给大家,具体如下:应用场景为什么要用redis?二进制存储、java序列化传输、IO连接数高
最近项目要用到redis,很多东西忘得差不多了,稍微回顾了利用Java客户端连接redis的过程,这里jedis是连接redis的Java客户端,如果没有Mav
本文实例讲述了java使用redis的方法。分享给大家供大家参考,具体如下:安装开始在java中使用redis前,我们需要确保已经安装了redis服务及java
如果redis已在线上业务使用中,但没有添加密码认证,那么如何在不影响业务服务的前提下给redis添加密码认证,就是一个需要仔细考虑的问题。本文描述一种可行的方