news 2026/3/7 1:01:01

Redis实现附近的人

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Redis实现附近的人

Redis 3.2版本以后,基于geohash和数据结构Zset提供了地理位置相关功能。通过上边两种mysql的实现方式发现,附近的人功能是明显的读多写少场景,所以用redis性能更会有很大的提升。

redis 实现附近的人功能主要通过Geo模块的六个命令。

GEOADD:将给定的位置对象(纬度、经度、名字)添加到指定的key;
GEOPOS:从key里面返回所有给定位置对象的位置(经度和纬度);
GEODIST:返回两个给定位置之间的距离;
GEOHASH:返回一个或多个位置对象的Geohash表示;
GEORADIUS:以给定的经纬度为中心,返回目标集合中与中心的距离不超过给定最大距离的所有位置对象;
GEORADIUSBYMEMBER:以给定的位置对象为中心,返回与其距离不超过给定最大距离的所有位置对象。

GEOADD key longitude latitude member [longitude latitude member ...] 如下: GEOADD hotel 119.98866180732716 30.27465803229662 酒馆

java中使用redis实现:

package com.demo.controller; import lombok.Data; import lombok.experimental.Accessors; import org.example.DemoRedisApplication; import org.junit.jupiter.api.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.data.geo.*; import org.springframework.data.redis.connection.RedisGeoCommands; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.data.redis.core.StringRedisTemplate; import java.util.ArrayList; import java.util.List; @SpringBootTest(classes = DemoRedisApplication.class) public class Test1 { @Autowired private StringRedisTemplate stringRedisTemplate; @Autowired private RedisTemplate<String, Object> redisTemplate; @Test public void test1() { //save(); List<UserInfo> userInfos = nearBySearch(100, 116.397428, 39.90923); userInfos.forEach(System.out::println); } //GEO相关命令用到的KEY private final static String KEY = "user_info"; public boolean save() { List<UserInfo> list = new ArrayList<>(); list.add(new UserInfo().setName("张三").setLatitude(39.90923).setLongitude(116.397428)); list.add(new UserInfo().setName("王五").setLatitude(39.90923).setLongitude(26.397428)); list.add(new UserInfo().setName("赵六").setLatitude(19.90923).setLongitude(106.397428)); list.add(new UserInfo().setName("小二").setLatitude(9.90923).setLongitude(116.397428)); list.add(new UserInfo().setName("小三").setLatitude(69.90923).setLongitude(96.397428)); list.add(new UserInfo().setName("小四").setLatitude(39.90923).setLongitude(116.397428)); list.add(new UserInfo().setName("小五").setLatitude(69.90923).setLongitude(116.397428)); list.add(new UserInfo().setName("小六").setLatitude(59.90923).setLongitude(116.397428)); list.add(new UserInfo().setName("小七").setLatitude(79.90923).setLongitude(116.397428)); list.add(new UserInfo().setName("小八").setLatitude(9.90923).setLongitude(116.397428)); list.add(new UserInfo().setName("小九").setLatitude(32.90923).setLongitude(106.397428)); for (UserInfo userInfo : list){ Long flag = redisTemplate.opsForGeo().add(KEY, new RedisGeoCommands.GeoLocation<>( userInfo.getName(), new Point(userInfo.getLongitude(), userInfo.getLatitude())) ); } return true; /*Long flag = redisTemplate.opsForGeo().add(KEY, new RedisGeoCommands.GeoLocation<>( userInfo.getName(), new Point(userInfo.getLongitude(), userInfo.getLatitude())) ); return flag != null && flag > 0;*/ } /** * 根据当前位置获取附近指定范围内的用户 * @param distance 指定范围 单位km ,可根据{@link org.springframework.data.geo.Metrics} 进行设置 * @param userLng 用户经度(-180,180) * @param userLat 用户纬度(-90,90) * @return */ public List<UserInfo> nearBySearch(double distance, double userLng, double userLat) { List<UserInfo> users = new ArrayList<>(); // 1.GEORADIUS获取附近范围内的信息 GeoResults<RedisGeoCommands.GeoLocation<Object>> reslut = redisTemplate.opsForGeo().radius(KEY, new Circle(new Point(userLng, userLat), new Distance(distance, Metrics.KILOMETERS)), RedisGeoCommands.GeoRadiusCommandArgs.newGeoRadiusArgs() .includeDistance() .includeCoordinates().sortAscending()); //2.收集信息,存入list List<GeoResult<RedisGeoCommands.GeoLocation<Object>>> content = reslut.getContent(); //3.过滤掉超过距离的数据 content.forEach(a-> { users.add( new UserInfo() .setName(a.getContent().getName().toString()) .setDistance(a.getDistance().getValue()) .setLatitude(a.getContent().getPoint().getX()) .setLongitude(a.getContent().getPoint().getY())); }); return users; } @Data @Accessors(chain = true) public static class UserInfo{ private String name;// 酒店 private double latitude;// 用户经度(-180,180) private double longitude;// 用户纬度(-90,90) private double distance;// 距离 } }

测试一下ok了,还可以使用 mongdb来实现。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/2/28 15:26:23

Java后端面试:原来大家也都是“半吊子”!

谁懂啊&#xff01;刚准备Java后端面试那会&#xff0c;我天天刷面经刷到凌晨&#xff0c;总觉得别人都把JVM、并发编程、分布式吃透了&#xff0c;就我是个“半吊子”。 直到面了3家公司我才发现——原来大家都是“半吊子”&#xff01; 整理了5个高频坑&#xff0c;备考的同…

作者头像 李华
网站建设 2026/3/6 20:11:16

基于微信小程序的在线家庭护理系统毕业设计源码

博主介绍&#xff1a;✌ 专注于Java,python,✌关注✌私信我✌具体的问题&#xff0c;我会尽力帮助你。一、研究目的本研究旨在设计并实现一个基于微信小程序的在线家庭护理系统&#xff0c;以解决当前家庭护理中存在的诸多问题。具体研究目的如下&#xff1a; 首先&#xff0c;…

作者头像 李华
网站建设 2026/3/4 22:35:41

基于微信小程序的校园电子课表系统毕设源码

博主介绍&#xff1a;✌ 专注于Java,python,✌关注✌私信我✌具体的问题&#xff0c;我会尽力帮助你。一、研究目的本研究旨在设计并实现一款基于微信小程序的校园电子课表系统&#xff0c;以满足现代高校学生对于便捷、高效、个性化的课程时间管理需求。具体研究目的如下&…

作者头像 李华
网站建设 2026/3/5 13:49:40

QM计划政策之工作篇核心要点解析

一、企业核岗问题处理机制当申报企业因岗位饱和等原因拒绝接收人才入职时&#xff0c;可通过以下路径解决:1.未入职状态解决方案:由属地人才部门协调&#xff0c;在同一行政区内更换接收企业。优势:保留省、市两级配套奖励资格。2.已入职状态解决方案:办理组织关系转移至新单位…

作者头像 李华
网站建设 2026/3/6 2:51:27

敏感词库自定义配置:LobeChat内容安全控制

敏感词库自定义配置&#xff1a;LobeChat内容安全控制 在企业开始将大语言模型&#xff08;LLM&#xff09;深度集成到客服、知识问答甚至内部协作系统中的今天&#xff0c;一个看似简单却极为关键的问题浮出水面&#xff1a;如何防止AI说出不该说的话&#xff1f; 想象这样一…

作者头像 李华
网站建设 2026/3/5 5:21:16

别再问 “要不要做等保” 了!一文讲清合规要求与企业风险

在数字化深入推进的今天&#xff0c;仍有不少企业纠结“是否需要做等保”。答案其实早已明确&#xff1a;等保不是可选项&#xff0c;而是企业运营的“必修课”。尤其2025年等保新规落地后&#xff0c;合规要求更趋严格&#xff0c;忽视等保不仅面临法律风险&#xff0c;更可能…

作者头像 李华