欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 新闻 > 社会 > 基于Redis实现-附近商铺查询

基于Redis实现-附近商铺查询

2025/5/6 21:20:19 来源:https://blog.csdn.net/2302_77423520/article/details/147653852  浏览:    关键词:基于Redis实现-附近商铺查询

基于Redis实现-附近查询

这个功能将使用到Redis中的GEO这种数据结构来实现。

1.GEO相关命令

GEO就是Geolocation的简写形式,代表地理坐标。Redis在3.2版本中加入到了对GEO的支持,允许存储地理坐标信息,帮助我们根据经纬度来检索数据,常见命令如下:

  • GEOADD: 添加一个地理空间信息,包含:经度(longitude)、纬度(latitude)、值(member)

  • GEODIST: 计算指定的两个点之间的距离并返回

  • GEOHASH: 将指定member的坐标转为hash字符串形式并返回

  • GEOPOS: 返回指定member的坐标

  • GEORADIUS: 指定圆心、半径,找到该圆内包含的所有member,并按照与圆心之间的距离排序后返回。6.2以后已废弃

  • GEOSEARCH: 在指定范围内搜索member,并按照与指定点之间的距离排序后返回。范围可以是圆形或矩形。6.2.新功能

  • GEOSEARCHSTORE: 与GEOSEARCH功能一致,不过可以把结果存储到一个指定的key。6.2.新功能

2.使用GEO来实现以下功能

  1. 添加下面几条数据:

    • 北京南站(116.378248 39.865275)
    • 北京站(116.42803 39.903738)
    • 北京西站(116.322287 39.893729)
    # 1. 添加地理空间数据
    GEOADD stations 116.378248 39.865275 "北京南站" 116.42803 39.903738 "北京站" 116.322287 39.893729 "北京西站"
    

    在这里插入图片描述

  2. 计算北京西站到北京站的距离

    # 2. 计算北京西站到北京站的距离
    GEODIST stations "北京西站" "北京站" km
    

    在这里插入图片描述

  3. 搜索天安门(116.397904 39.909005)附近10km内的所有火车站,并按照距离升序排序

    # 3. 搜索天安门附近10km内的火车站并按距离排序
    GEOSEARCH stations FROMLONLAT 116.397904 39.909005 BYRADIUS 10 km ASC
    

    在这里插入图片描述

在这里插入图片描述

3.使用Java实现简单的附近商铺查询

//在ServiceImpl中(简单演示)
@Autowiredprivate StringRedisTemplate stringRedisTemplate;
@Override
public Result queryShopByType(Integer typeId, Integer current, Double x, Double y) {// 1. 判断是否需要基于地理位置查询if (x == null || y == null) {// 不需要地理坐标查询时,直接按类型分页查询数据库Page<Shop> page = query().eq("type_id", typeId)  // 按店铺类型筛选.page(new Page<>(current, SystemConstants.DEFAULT_PAGE_SIZE));  // 分页查询// 返回查询结果return Result.ok(page.getRecords());}// 2. 需要地理查询时,计算分页参数int from = (current - 1) * SystemConstants.DEFAULT_PAGE_SIZE;  // 起始偏移量int end = current * SystemConstants.DEFAULT_PAGE_SIZE;          // 结束位置// 3. 从Redis中查询附近店铺ID(GEO查询)String key = "shop:geo:" + typeId;  // GEO数据存储的key// 执行GEO搜索:以(x,y)为中心,5000米半径范围内,查询end数量的店铺GeoResults<RedisGeoCommands.GeoLocation<String>> results = stringRedisTemplate.opsForGeo().search(key,GeoReference.fromCoordinate(x, y),  // 中心点坐标new Distance(5000),                // 搜索半径(5公里)RedisGeoCommands.GeoSearchCommandArgs.newGeoSearchArgs().includeDistance()         // 包含距离信息.limit(end)                // 限制返回数量);// 4. 处理查询结果并获取店铺详情if (results == null) {return Result.ok(Collections.emptyList());  // 无结果时返回空列表}List<GeoResult<RedisGeoCommands.GeoLocation<String>>> list = results.getContent();if (list.size() <= from) {return Result.ok(Collections.emptyList());  // 结果不足时分页返回空}// 收集店铺ID和距离信息List<Long> ids = new ArrayList<>(list.size());Map<String, Distance> distanceMap = new HashMap<>(list.size());// 跳过前from条记录(分页处理),然后处理剩余记录list.stream().skip(from).forEach(result -> {String id = result.getContent().getName();  // 获取店铺IDids.add(Long.valueOf(id));distanceMap.put(id, result.getDistance()); // 存储店铺距离});// 根据ID批量查询店铺详情(保持ID顺序)String strIds = StrUtil.join(",", ids);List<Shop> shops = query().in("id", ids)  // 按ID列表查询.last("ORDER BY FIELD(id," + strIds + ")")  // 保持Redis返回的顺序.list();// 为每个店铺设置距离信息for (Shop shop : shops) {shop.setDistance(distanceMap.get(shop.getId().toString()).getValue());}// 5. 返回带距离信息的店铺列表return Result.ok(shops);
}

4.Redis GEO search 方法的参数

  1. 排序方式
.sortAscending()    // 按距离升序排序(从近到远)
.sortDescending()  // 按距离降序排序(从远到近)
  1. 返回内容控制
.includeDistance()  // 在结果中包含距离信息
.includeCoordinates() // 在结果中包含坐标信息
.includeName()     // 在结果中包含成员名称(默认包含)
  1. 结果限制
.limit(50)         // 限制返回结果数量(可用于简单分页)
  1. 单位设置
.includeDistance().withDistance(Metrics.KILOMETERS) // 指定距离单位
//支持的单位:
//Metrics.KILOMETERS(千米)
//Metrics.MILES(英里)
//Metrics.FEET(英尺)
//Metrics.METERS(米)
  1. GeoReference 的三种主要形式
//1.fromCoordinate(x, y)
//作用:通过经纬度坐标指定中心点
//示例:
GeoReference.fromCoordinate(116.404, 39.915)  // 北京天安门坐标//2.fromMember(memberName)
//作用:通过 Redis中已存储的GEO成员名称指定中心点
//示例:
GeoReference.fromMember("北京站")  // 以已存储的"北京站"坐标为中心//3.fromString("x,y")
//作用:通过字符串格式的坐标指定中心点
//示例:
GeoReference.fromString("116.404,39.915")

如下为完整示例:

GeoResults<RedisGeoCommands.GeoLocation<String>> results = stringRedisTemplate.opsForGeo().search("shop:geo:1",GeoReference.fromCoordinate(116.397904, 39.909005),new Distance(5, Metrics.KILOMETERS),RedisGeoCommands.GeoSearchCommandArgs.newGeoSearchArgs().includeDistance()      // 包含距离.includeCoordinates()   // 包含坐标.sortAscending()        // 按距离升序.limit(100)             // 最多返回100条.withDistance(Metrics.KILOMETERS) // 距离单位为千米
);

对于 Redis 6.2 及以上版本,还可以使用:

  1. 矩形范围搜索
.byBox(width, height, Metrics.KILOMETERS) // 矩形范围搜索
  1. 存储搜索结果
.store("result-key")  // 将结果存储到指定key
.storeDist("result-key") // 存储带距离的结果

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词