附近商户-GEO数据结构的基本用法

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

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

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

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

  • GEOPOS:返回指定member的坐标

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

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

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

导入店铺数据到GEO

![[Pasted image 20250718204058.png]]

当我们点击美食之后,会出现一系列的商家,商家中可以按照多种排序方式,我们此时关注的是距离,这个地方就需要使用到我们的GEO,向后台传入当前app收集的地址(我们此处是写死的) ,以当前坐标作为圆心,同时绑定相同的店家类型type,以及分页信息,把这几个条件传入后台,后台查询出对应的数据再返回。

我们把x轴坐标和y轴坐标当作score传入到redis中,不要把所有数据都放到redis
同时把商户类型做分类,相同类型的商户作为一组,放入同一个GEO集合中
![[Pasted image 20250718204424.png]]

@Test
void loadShopData() {// 1.查询店铺信息List<Shop> list = shopService.list();// 2.把店铺分组,按照typeId分组,typeId一致的放到一个集合Map<Long, List<Shop>> map = list.stream().collect(Collectors.groupingBy(Shop::getTypeId));// 3.分批完成写入Redisfor (Map.Entry<Long, List<Shop>> entry : map.entrySet()) {// 3.1.获取类型idLong typeId = entry.getKey();String key = SHOP_GEO_KEY + typeId;// 3.2.获取同类型的店铺的集合List<Shop> value = entry.getValue();List<RedisGeoCommands.GeoLocation<String>> locations = new ArrayList<>(value.size());// 3.3.写入redis GEOADD key 经度 纬度 memberfor (Shop shop : value) {// stringRedisTemplate.opsForGeo().add(key, new Point(shop.getX(), shop.getY()), shop.getId().toString());locations.add(new RedisGeoCommands.GeoLocation<>(shop.getId().toString(),new Point(shop.getX(), shop.getY())));}stringRedisTemplate.opsForGeo().add(key, locations);}
}

先查询店铺信息,从数据库中获取所有店铺的信息,并将其存储到List类型的变量list中
按店铺类型分组,利用Stream API ,代码将获取的店铺列表按typeId分组,相同typeId的店铺会被归于同一个列表,最终形成Map<Long,List>类型的映射map,其中键是typeId,值是对应类型的店铺列表。
对于每个类型的店铺,先获取当前类型的typeId,创建对应的Redis Key值
对于每个店铺,创建一个RedisGeoCommands.GeoLocation对象,包括店铺ID、经纬度等
将所有对象收集到一个列表,最后通过opsForGeo()批量写入redis

附近商户-实现附近商户功能

   // 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());}

先看有没有传坐标,没有直接按数据库查询

    // 3.查询redis、按照距离排序、分页。结果:shopId、distanceString key = SHOP_GEO_KEY + typeId;GeoResults<RedisGeoCommands.GeoLocation<String>> results = stringRedisTemplate.opsForGeo() // GEOSEARCH key BYLONLAT x y BYRADIUS 10 WITHDISTANCE.search(key,GeoReference.fromCoordinate(x, y),new Distance(5000),RedisGeoCommands.GeoSearchCommandArgs.newGeoSearchArgs().includeDistance().limit(end));

若传输了坐标,就按距离排序后分页

  • stringRedisTemplate.opsForGeo().search()调用 Redis 的 GEOSEARCH 命令。
  • key:指定要搜索的地理位置数据集合。
  • GeoReference.fromCoordinate(x, y):指定中心点坐标。
  • new Distance(5000):设置搜索半径为 5000 米。
  • includeDistance().limit(end):要求返回距离信息,并限制最多返回end条结果。
 List<Long> ids = new ArrayList<>(list.size());Map<String, Distance> distanceMap = new HashMap<>(list.size());list.stream().skip(from).forEach(result -> {// 4.2.获取店铺idString shopIdStr = result.getContent().getName();ids.add(Long.valueOf(shopIdStr));// 4.3.获取距离Distance distance = result.getDistance();distanceMap.put(shopIdStr, distance);});

创建ids列表存放店铺ID
常见distanceMap存储店铺ID和距离的映射关系
借助流中的skip(from)跳过前from个元素
然后对于每个元素获取店铺id和距离,添加到ids和distanceMap中

    // 5.根据id查询ShopString idStr = StrUtil.join(",", ids);List<Shop> shops = query().in("id", ids).last("ORDER BY FIELD(id," + idStr + ")").list();for (Shop shop : shops) {shop.setDistance(distanceMap.get(shop.getId().toString()).getValue());}// 6.返回return Result.ok(shops);

将ids拼成字符串然后通过mybatis-plus提供的数据库查询操作得到店铺,对于每一个店铺,利用之前得到的distanceMap将位置信息赋值

总结

  1. Redis GEO 数据结构在 “附近商户” 功能中核心作用是什么?
    Redis GEO 用于存储地理坐标信息(经度、纬度),并提供高效的地理位置查询能力(如范围搜索、距离计算),是实现 “按距离筛选附近商户” 功能的核心存储与查询工具。
  2. 商户数据导入 Redis GEO 的关键步骤是什么?
    核心步骤包括:①从数据库查询所有商户信息;②按商户类型(typeId)分组,确保同类型商户聚合;③将每组商户的经纬度、ID 封装为 GEO 坐标对象,批量写入对应类型的 Redis GEO 集合(通过GEOADD命令)。
  3. 实现 “附近商户” 查询的核心流程是什么?
    ①接收用户当前坐标、商户类型、分页参数;②通过 Redis GEO 的GEOSEARCH命令,以用户坐标为中心,按指定半径查询目标类型商户的 ID 及距离;③根据 ID 从数据库查询商户详细信息,并关联 Redis 返回的距离数据;④返回带距离的商户列表。
  4. 为何要按商户类型分组存储 GEO 数据?
    按类型分组可缩小查询范围,避免无关类型商户参与计算,提升GEOSEARCH命令的执行效率;同时符合用户 “按类型找商户” 的实际业务场景,减少无效数据处理。
  5. 如何保证查询结果中 “距离” 信息的准确性?
    Redis GEO 在范围查询时通过includeDistance参数返回商户与中心点的距离,查询后将该距离与数据库查询的商户信息一一映射(通过商户 ID 关联),确保距离与商户信息精准匹配。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/915035.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/915035.shtml
英文地址,请注明出处:http://en.pswp.cn/news/915035.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【vue-3】深入理解 Vue 3 中的 v-if 指令:条件渲染的艺术

在 Vue.js 的世界中&#xff0c;条件渲染是构建动态界面的核心概念之一。作为 Vue 3 中最常用的指令之一&#xff0c;v-if 提供了强大的能力来控制元素的显示与隐藏。本文将深入探讨 v-if 的工作原理、最佳实践以及它在 Vue 3 中的新特性。 1. 什么是 v-if&#xff1f; v-if 是…

【实时Linux实战系列】实时系统中的内存策略

在实时系统中&#xff0c;内存管理是确保系统性能和稳定性的重要组成部分。实时系统通常需要快速响应和低延迟&#xff0c;因此高效的内存管理策略对于实现这些目标至关重要。实时 Linux 提供了多种内存管理机制&#xff0c;如使用大型页面&#xff08;Huge Pages&#xff09;和…

【C语言进阶】题目练习(2)

目录 题目6:看代码说结果 分析&#xff1a; 答案&#xff1a;255 题目7&#xff1a;猜名次 分析&#xff1a; 题目8&#xff1a;猜凶手 分析&#xff1a; 代码&#xff1a; 题目9&#xff1a;打印杨辉三角 思路: 代码: 题目10&#xff1a;关于指针的选择题 答案&a…

思科NAT综合实验

1 实验拓扑图2实验目的(1)巩固前面实验的配置(2)掌握四种NAT的配置(3)明白四种NAT的区别3实验步骤3.1配置边界路由器和外网路由器的端口IP三个步骤&#xff1a;进入端口 打开端口 配置IP地址和子网掩码interface f0/0 no shutdown ip address 192.168.201.2 255.255.255.03.2配…

VMC850立式加工中心Y轴传动机械结构设计cad【7张】三维图+设计说明书

摘 要 数控机床作为现代工业生产的重要设备&#xff0c;对国民经济的发展有着重要的作用&#xff0c;立式加工中心作为数控加工技术的核心&#xff0c;通过对其研究&#xff0c;可以深入了解数控技术未来的发展方向。本文主要完成了VMC850立式加工中心Y轴的机械传动结构设计&am…

mpiigaze的安装过程一

mpiigaze链接 mpiigaze应该不是作者本人写的&#xff0c;而是社区工作者的杰作&#xff0c;对原论文Appearance-Based Gaze Estimation in the Wild的代码进行的一些复现 1.创建conda环境 2.问题 Building wheels for collected packages: dlibBuilding wheel for dlib (py…

如何将华为文件传输到电脑

在数字管理领域&#xff0c;将华为设备上的文件传输到电脑是高频需求。无论为了备份、缓解手机存储压力&#xff0c;还是跨平台访问&#xff0c;把华为手机连接电脑已成为许多用户的刚需。下面介绍 5 种高效方法&#xff0c;可满足不同场景与偏好&#xff0c;助你轻松完成文件迁…

LP-MSPM0G3507学习--05中断及管脚中断

关键函数&#xff1a; NVIC_EnableIRQ(IRQn_Type IRQn)&#xff1a;使能中断 例5-1&#xff1a;单按键中断方式实现led灯的亮灭 在上一讲LP-MSPM0G3507学习--04GPIO控制中实现了通过按键控制led灯的亮灭&#xff0c;可以看出程序效率不高&#xff0c;下面采用中断的方式实现…

mac系统安装、启动Jenkins,创建pytest接口自动化任务

先安装Homebrew&#xff1a;mac系统安装brew-CSDN博客 1、安装Jenkins # 可以安装长期支持版本 brew install jenkins-lts# 或者最新版本&#xff08;我安了这个&#xff09; brew install jenkins 可查看Jenkins安装位置&#xff1a; # 最新版本 brew --prefix jenkins 2、…

设置第三方窗口置顶(SetWindowPos方法,vb.net)

起源在日常办公、游戏时&#xff0c;我们经常需要一些窗口处于置顶状态&#xff0c;而这些窗口往往是网页端&#xff08;浏览器&#xff09;、办公软件&#xff08;wps、office等&#xff09;&#xff0c;这些需要置顶的窗口往往自身没有明显的置顶开关&#xff0c;因此&#x…

Docker-下载和安装

一、Linux版 1.安装docker &#xff08;1&#xff09;更新软件包索引 sudo apt update &#xff08;2&#xff09;安装必要的依赖 sudo apt install apt-transport-https ca-certificates curl software-properties-common &#xff08;3&#xff09;添加 Docker 官方 GP…

电脑DLL错误修复dll微软运行库工具修复dll缺失找不到dll等问题,dll免费修复工具

解决DLL文件缺失问题&#xff1a;我的使用体验与建议 在使用电脑的过程中&#xff0c;我们常常会遇到软件或系统报错&#xff0c;例如“无法找到指定模块”或“缺少某.dll文件”等提示。DLL&#xff08;动态链接库&#xff09;是Windows系统中不可或缺的组件&#xff0c;为应用…

HTTPS的工作原理及DNS的工作过程

HTTPSHTTP协议安全上存在以下三个风险&#xff1a;完整性 可用性 保密性窃听风险&#xff0c;比如通信链路上可以获取通信内容&#xff0c;用户号容易没。篡改风险&#xff0c;比如强制植入垃圾广告&#xff0c;视觉污染&#xff0c;用户眼容易瞎。冒充风险&#xff0c;比如冒充…

VisualXML全新升级 | 新增BusLoad计算

VisualXML是一个功能强大的网络总线设计工具&#xff0c;专注于简化汽车电子系统中复杂的网络数据设计操作。该软件支持多种主流总线网络格式的数据编辑&#xff08;如DBC、LDF、ARXML、HEX等&#xff09;&#xff0c;并能够基于Excel表格的方式生成和转换多种数据库文件。由此…

李天意考研数学精讲课学习笔记(课堂版)

视频链接&#xff1a;【考研数学精讲课李天意】基础强化真题&#xff0c;概念精讲与解题技巧&#xff08;适用数学一/二/三&#xff09;_哔哩哔哩_bilibili 讲义&#xff1a;夸克网盘分享 高数6 不定积分

闲庭信步使用图像验证平台加速FPGA的开发:第二十三课——图像直方图和灰度图像叠加的FPGA实现

&#xff08;本系列只需要modelsim即可完成数字图像的处理&#xff0c;每个工程都搭建了全自动化的仿真环境&#xff0c;只需要双击top_tb.bat文件就可以完成整个的仿真&#xff0c;大大降低了初学者的门槛&#xff01;&#xff01;&#xff01;&#xff01;如需要该系列的工程…

C++并发编程-14. 利用栅栏实现同步

前文我们通过原子操作实战实现了无锁队列&#xff0c;今天完善一下无锁的原子操作剩余的知识&#xff0c;包括Relaese和Acquire内存序在什么情况下是存在危险的&#xff0c;以及我们可以利用栅栏机制实现同步等等。 线程可见顺序 我们提到过除了memory_order_seq_cst顺序&#…

如何选择旅游科技行业云ERP?Oracle NetSuite助力汇智国际数智化升级

2025年4月21日&#xff0c;汇智国际旅游发展有限公司&#xff08;以下简称汇智国际&#xff09;携手 Oracle NetSuite与Hitpoint Cloud &#xff0c;共同参与了汇智国际 Oracle NetSuite 云ERP 项目启动会。 本次会议标志着汇智国际在数字化转型道路上迈出了坚实而关键的一步&…

深度学习零基础入门(3)-图像与神经网络

好久不见~我又回来了 这一节我们来讲一讲图像在计算机中的本质&#xff0c;以及全连接神经网络的缺陷&#xff0c;进而引出卷积神经网络一、图像在计算机中的本质 不知道你有没有学过数据结构&#xff0c;在讲这一部分的时候对数组进行了扩展&#xff0c;讲到了广义表和压缩矩阵…

http性能测试命令ab

在 Linux系统中&#xff0c; ab&#xff08; ApacheBench&#xff09;是一个用于 测试HTTP服务器性能的 工具。它是 Apache HTTP服务器项目的 一部分&#xff0c;专门设计用来模拟 多个用户对 服务器发起 并发请求&#xff0c;从而 评估服务器的 负载能力和 响应时间其中&#…