当前位置: 首页 > news >正文

基于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") // 存储带距离的结果

相关文章:

  • 在项目中如何对Map List等对象序列化及反序列化
  • 第二章 OpenCV篇-图像阈值的综合操作-Python
  • 【瑞萨RA4L1-SENSOR套件测评】LCD 实时显示 ADC 水位高度
  • vue中$set原理
  • AIO-1126-JD4蓝牙通讯(BLE)
  • Lucene并不是只有倒排索引一种数据结构,支持多种数据结构
  • Angular教程前言:历史、安装与用途
  • Webug4.0靶场通关笔记08- 第11关万能密码登录(SQL注入漏洞)
  • 数字智慧方案5968丨智慧农业产销一体化云平台建设方案(72页PPT)(文末有下载方式)
  • 【Docker】Dockerfile 使用
  • python如何把pdf转word
  • 高等数学-第七版-下册 选做记录 习题9-7
  • C++11新特性_自动类型推导_auto
  • 【RocketMQ】- 源码系列目录
  • 神经网络—损失函数
  • Windows服务器提权实战:常见方法、场景与防御指南
  • 2025五一杯C题五一杯数学建模思路代码文章教学:社交媒体平台用户分析问题
  • 【精选】基于数据挖掘的广州招聘可视化分析系统(大数据组件+Spark+Hive+MySQL+AI智能云+DeepSeek人工智能+深度学习之LSTM算法)
  • 【playwright】内网离线部署playwright
  • UE实用地编插件Physical Layout Tool
  • 党旗下的青春|赵天益:少年确定志向,把最好的时光奉献给戏剧事业
  • 张建华评《俄国和法国》|埃莲娜·唐科斯的俄法关系史研究
  • 韩国检方结束对尹锡悦私宅的扣押搜查
  • 武汉楼市新政:二孩、三孩家庭购买新房可分别享受6万元、12万元购房补贴
  • 奈雪的茶叫停“能喝奶茶就不要喝水”宣传,当地市监称不要误导消费者
  • 医学统计专家童新元逝世,终年61岁