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

【redis实战篇】第八天

摘要:

        本文主要介绍redis中GEO和BitMap结构的基本用法和用处,并基于这两种结构实现java项目黑马点评的实际功能--根据距离查询附近商铺以及签到和签到统计

 

一,根据距离查询商铺功能

1,GEO介绍

GEO(地理空间)结构是一种用于存储地理坐标数据,并支持基于地理位置的查询功能的数据类型。其本质上是通过有序集合(ZSET)实现的,支持附近位置查询、距离计算等操作

常用命令:

(1)GEOADD key longitude latitude member [longitude latitude member ...] 

(2)GEOPOS key member [member ...] 查询成员的经纬度坐标,返回数组形式的结果

(3)GEODIST key member1 member2 [unit] 计算两个坐标点之间的距离

(4)GEORADIUS key longitude latitude radius unit [WITHCOORD] [WITHDIST] [WITHHASH] [COUNT count] 以指定坐标为中心,查询半径内的所有点,支持返回距离、坐标等信息

2,代码实现

(1)redis中插入数据,将商铺类型作为key,商铺id作为member,使用stearm流将将商店根据类型进行分组得到map<类型id,list<商铺实体>>,最后构造locations数组批量写入

List<Shop> list = shopService.list();
//按照typeId分组
Map<Long, List<Shop>> map = list.stream().collect(Collectors.groupingBy(Shop::getTypeId));
for (Map.Entry<Long, List<Shop>> entry : map.entrySet()) {Long typeId = entry.getKey();List<Shop> shops = entry.getValue();String key = "shop:geo:" + typeId;List<RedisGeoCommands.GeoLocation<String>> locations = new ArrayList<>(shops.size());//将分组结果写入redis  geoadd key  longitude latitude lo la.... typeId//list集合数据放入locations集合中shop-->GeoLocation(name,point)for (Shop shop : shops) {locations.add(new RedisGeoCommands.GeoLocation<>(shop.getId().toString(),new Point(shop.getX(),shop.getY())));}//批量写入stringRedisTemplate.opsForGeo().add(key,locations);
}

(2)如果前端携带的经纬坐标为空,执行传统的数据库查询(分页)

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)根据默认的页数5计算分页参数(起始),通过前端携带的经纬坐标xy为圆心,距离默认5km,添加返回条件--商铺到xy距离以及limit范围

        int from = (current - 1) * SystemConstants.DEFAULT_PAGE_SIZE;int end = current * SystemConstants.DEFAULT_PAGE_SIZE;//查询redisString key = SHOP_GEO_KEY + typeId;//limit默认查询从0-end,需要手动截取GeoResults<RedisGeoCommands.GeoLocation<String>> results = stringRedisTemplate.opsForGeo().search(key,//圆心GeoReference.fromCoordinate(x, y),//半径new Distance(5000),//返回条件--返回距离和上限RedisGeoCommands.GeoSearchCommandArgs.newGeoSearchArgs().includeDistance().limit(end));if (results==null) {return Result.ok(Collections.emptyList());}

(4)拿到0-end部分的数据(包括距离、member和point坐标),同时判断是否还存在下一页,不存在则返回空集合

List<GeoResult<RedisGeoCommands.GeoLocation<String>>> list = results.getContent();
//判断是否还有下一页
if (list.size()<=from) {return Result.ok(Collections.emptyList());
}

(5)将from到end部分的数据通过ids商铺id集合和map每个商铺对应距离集合收集,使用stream流的skip跳过from前的数据,保证数据为当前页

        //截取from-end部分,ids和map收集起来List<Long> ids = new ArrayList<>(list.size());Map<String, Distance> distanceMap = new HashMap<>(list.size());//skip跳过手动截取list.stream().skip(from).forEach(result->{//获取店铺id和距离String shopId = result.getContent().getName();Distance distance = result.getDistance();//提取ids.add(Long.valueOf(shopId));distanceMap.put(shopId, distance);});

(6)批量查询商铺信息,并遍历所有商铺添加对应的距离参数,返回符合条件商铺结果

        //根据ids批量查询shopString join = StrUtil.join(",",ids);List<Shop> shops = query().in("id", ids).last("order by field(id," + join + ")").list();for (Shop shop : shops) {shop.setDistance(distanceMap.get(shop.getId().toString()).getValue());}

二,用户签到功能

1,bitMap简单介绍

BitMap 本质是字符串(String),但按位(bit)而非字节(byte)操作,每个位存储 0 或 1(布尔值)

 常用命令:

(1)SETBIT key offset value:设置偏移量offset的位为value(0 或 1)

(2)GETBIT key offset:获取偏移量offset的位值

(3)BITCOUNT key [start end]:统计指定范围内置为 1 的位数

2,签到实现

(1)原理:将每个用户这个月的签到情况使用bitMap表示,签到置1,未签到默认0,用户id拼接该月日期作为key,根据签到情况添加value

(2)因为偏移量是逻辑位置,所以存入时要对今天为该月天数减一

Long userId = UserHolder.getUser().getId();
LocalDateTime now = LocalDateTime.now();
//拼接key
String keySuffix = now.format(DateTimeFormatter.ofPattern(":yyyyMM"));
String key = USER_SIGN_KEY + userId + keySuffix;
//今天是这个月的第几天
int dayOfMonth = now.getDayOfMonth();
//存入redis的bitMap中,第一天存在第0位
Boolean success = stringRedisTemplate.opsForValue().setBit(key, dayOfMonth - 1, true);
return Result.ok("恭喜签到成功!");

3,统计连续签到天数(0111-->3天

(1)取出当前用户该月签到情况返回的十进制数num

        List<Long> result = stringRedisTemplate.opsForValue().bitField(key,BitFieldSubCommands.create().get(BitFieldSubCommands.BitFieldType.unsigned(dayOfMonth)).valueAt(0));if (result==null||result.isEmpty()) {return Result.ok(0);}//取出bitMap的十进制数Long num = result.getFirst();if (num==null||num==0) {return Result.ok(0);}

(2)从后遍历每个bit位-->使这个数与1做与运算,如果结果为1则累加天数count,将num右移一位并赋值给原num,否则认为签到中断break。

        int count = 0;while (true) {if ((num&1)==0) {break;}else {count++;}//右移一位并赋值给原numnum >>>= 1;}

http://www.lqws.cn/news/177967.html

相关文章:

  • 越狱蒸馏-可再生安全基准测试
  • Science Robotics:UCLA 贺曦敏团队综述自主软体机器人
  • 绕过 Xcode?使用 Appuploader和主流工具实现 iOS 上架自动化
  • AI大模型学习三十三、HeyGem.ai 服务端(ubuntu)docker 安装 /客户端(win)分离部署
  • 【人工智能】神经网络的优化器optimizer(二):Adagrad自适应学习率优化器
  • 智能仓储的未来:自动化、AI与数据分析如何重塑物流中心
  • vscode使用系列之快速生成html模板
  • LlamaFactory × 多模态RAG × Chat-BI:万字长文探寻RAG进化轨迹,打造卓越专业AI助手
  • 安卓基础(ProGuard vs R8)
  • FART 脱壳某大厂 App + CodeItem 修复 dex + 反编译还原源码
  • 【Linux】Linux 进程间通讯-管道
  • gitlab CI/CD本地部署配置
  • WebRTC 与 WebSocket 的关联关系
  • 【JVM】Java虚拟机(一)——内存结构
  • Qt生成日志与以及报错文件(mingw64位,winDbg)————附带详细解说
  • 在Windows下利用LoongArch-toolchain交叉编译Qt
  • 【PmHub面试篇】PmHub中基于Redis加Lua脚本的计数器算法限流实现面试专题解析
  • 数据库SQLite基础
  • Ubuntu18.6 学习QT问题记录以及虚拟机安装Ubuntu后的设置
  • 【Qt】:设置新建类模板
  • C/C++ 面试复习笔记(4)
  • Excel 发现此工作表中有一处或多处公式引用错误。请检查公式中的单元格引用、区域名称、已定义名称以及到其他工作簿的链接是否均正确无误。弹窗
  • 关于iview组件中使用 table , 绑定序号分页后序号从1开始的解决方案
  • excel数据对比找不同:6种方法核对两列数据差异
  • 天机学堂(学习计划和进度)
  • 内容力重塑品牌增长:开源AI大模型驱动下的智能名片与S2B2C商城赋能抖音生态种草范式
  • ESP8266(NodeMcu)+GPS模块+TFT屏幕实现GPS码表
  • 【PhysUnits】16.1 完善Var 结构体及其运算(variable.rs)
  • 多种风格导航菜单 HTML 实现(附源码)
  • 32单片机——基本定时器