springboot+redis+mybatis体会布隆过滤器

news2024/9/24 9:26:52

1.建立数据库表和对应实体类

CREATE TABLE `user` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `uname` varchar(50) DEFAULT NULL,
  `usex` varchar(20) DEFAULT NULL,
  `uage` int(11) DEFAULT NULL,
  PRIMARY KEY (`id`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=1319 DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC;
public class User {
    private String id;
    private String uname;
    private String usex;
    private String uage;

    public String getUname() {
        return uname;
    }

    public void setUname(String uname) {
        this.uname = uname;
    }

    public String getUsex() {
        return usex;
    }

    public void setUsex(String usex) {
        this.usex = usex;
    }

    public String getUage() {
        return uage;
    }

    public void setUage(String uage) {
        this.uage = uage;
    }

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }
}

2.编写Controller,新增用户和获取用户

@Controller
public class BloomController {
    @Autowired
    private BloomService bloomService;

    @RequestMapping(value = "/addUser")
    @ResponseBody
    public void addUser(){
        for(int i=0;i<100;i++){
            User user = new User();
            user.setUsex(i + "");
            user.setUname("user:" + i);
            user.setUage("26");
            bloomService.addUser(user);
        }

    }



    @RequestMapping(value = "/getUser")
    @ResponseBody
    public User getUser(HttpServletRequest request){
        String userId = request.getParameter("userId");
        User user = bloomService.getUser(userId);
        return  user;

    }

}

3.编写servie,实现具体的业务逻辑

public interface BloomService {
    void addUser(User user);

    User getUser(String userId);

    List<String> getUserIds();
}
@Service
public class BloomServiceImpl implements BloomService {

    public static final String redisKey = "USER:";

    @Resource
    private BloomDao bloomDao;

    @Autowired
    private RedisTemplate redisTemplate;

    @Autowired
    private BloomCheckUtil bloomCheckUtil;

    @Override
    public void addUser(User user) {
        int i = bloomDao.addUser(user);
        if(i>0){
            //mysql插入成功,将数据存入redis
            User currentUser = bloomDao.getUser(user.getId());
            redisTemplate.opsForValue().set(redisKey+user.getId(), JSONObject.toJSONString(currentUser));
        }


    }

    @Override
    public User getUser(String userId) {
        User user = null;
        //1.先从布隆过滤器中获取数据,若没有则直接返回
        if(!bloomCheckUtil.check(userId)){
            System.out.println("该用户id:" + userId +"不在合法用户内!!!!");
            return user;
        }
        //2.从redis中查找
        String rediskey = redisKey + userId;
        String jsonObj = Optional.ofNullable(redisTemplate.opsForValue().get(rediskey)).map(Object::toString).orElse(null);

         if(jsonObj == null){
             //3.redis中没有,查询数据库
             user = bloomDao.getUser(userId);
             if(user != null){
                 //4.redis无mysql有,把查询出的数据写入到redis缓存
                 redisTemplate.opsForValue().set(rediskey,JSONObject.toJSONString(user));
             }

             return user;
         }

        user =JSONObject.parseObject(jsonObj, User.class) ;
        return user;
    }

    @Override
    public List<String> getUserIds() {

        return bloomDao.getUserIds();
    }
}

4.编写dao和mapper,实现数据库操作

public interface BloomDao {
    int addUser(@Param("user") User user);

    User getUser(@Param("id")String id);

    List<String> getUserIds();
}
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" >
<mapper namespace="com.cn.study.redis.dao.BloomDao">

    <insert id="addUser"  parameterType="com.cn.study.redis.pojo.User" useGeneratedKeys="true" keyProperty="id">
        insert into user(uname,usex,uage) values (#{user.uname},#{user.usex},#{user.uage})
    </insert>

    <select id="getUser" resultType="com.cn.study.redis.pojo.User">
       select u.id ,u.uname ,u.usex ,u.uage  from user u where u.id = #{id}
    </select>

    <select id="getUserIds" resultType="String">
        select id from user
    </select>
</mapper>

5.编写布隆过滤器和工具类

@Component
public class BloomFilter {

    public static final String key = "LEGITIMATE_USER";

    @Autowired
    private RedisTemplate redisTemplate;

    @Autowired
    private BloomService bloomService;


    @PostConstruct
    public void init(){
        //初始化布隆过滤器
        //1.从数据库查出user表中的所有id,计算hash值
        List<String> userIdList = bloomService.getUserIds();

        for(int i=0;i<userIdList.size();i++){
            //对每个userId计算hash值并取绝对值
            int hashValue = Math.abs(("USER:" + userIdList.get(i)).hashCode());
            //使用hash值对2的32次方取余,得到下标
            long index = (long)(hashValue % Math.pow(2,32));

            System.out.println("id为:" + userIdList.get(i)+"的用户产生的hash值为:" + hashValue +",下标为:" + index);
            //将redis中对应下标的bitmap的值更新成1
            redisTemplate.opsForValue().setBit(key,index,true);
        }
    }
}
@Component
public class BloomCheckUtil {


    private static final String key = "LEGITIMATE_USER";

    @Autowired
    private RedisTemplate redisTemplate;

    public boolean check(String userId){
        boolean flag = false;

        int hashValue = Math.abs(("USER:" + userId).hashCode());

        long index = (long)(hashValue % Math.pow(2,32));

        flag = redisTemplate.opsForValue().getBit(key,index);

        return flag;

    }

}

7.验证

由于第一次启动项目的时候,BloomFilter 类中的init方法会在一开始执行,此时我还并没有调用addUser方法添加用户,数据库中没有任何用户信息,所以第一次启动项目的时候,布隆过滤器中不会有任何的数据。
我们启动项目后,手动调用下addUser方法,新增100条用户数据。
然后重启项目,将用户信息加载到布隆过滤器中。
我们调用getUser方法尝试获取用户信息,当我们传了一个数据库不存在的用户id时,当请求到布隆过滤器,布隆过滤器发现没有此数据,则直接返回,不会再往下查redis和mysql
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

美国硅谷多ip服务器用途广吗?

美国硅谷多IP服务器的用途广泛&#xff0c;涉及多个行业和应用场景&#xff0c;包括站群运营、SEO优化、游戏代理、软件开发、数据分析、科学研究、电子商务、在线营销、虚拟主机和云服务等。具体分析如下&#xff0c;rak小编为您整理发布美国硅谷多ip服务器用途广吗的相关内容…

JetBrains WebStorm 2024.2 (macOS, Linux, Windows) - 最智能的 JavaScript IDE

JetBrains WebStorm 2024.2 (macOS, Linux, Windows) - 最智能的 JavaScript IDE JetBrains 跨平台开发者工具 请访问原文链接&#xff1a;ttps://sysin.org/blog/jetbrains-webstorm/&#xff0c;查看最新版。原创作品&#xff0c;转载请保留出处。 作者主页&#xff1a;sy…

OpenCV绘图函数(12)绘制直线函数 line()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 line 函数在图像中绘制从 pt1 到 pt2 两点之间的线段。这条线会被图像边界裁剪。对于没有抗锯齿效果且坐标为整数的线&#xff0c;会使用 8-连接…

HarmonyOS开发实战( Beta5版)减小应用包大小

简介 减小应用包大小是提升应用下载、安装体验的重要方式之一。通过压缩、精简或者复用应用中的代码或资源&#xff0c;可以有效降低应用的大小&#xff0c;提升应用下载和安装速度&#xff0c;减少系统空间占用。 开发者可以参考下面三种方法减小应用包大小&#xff1a; 配…

数据结构—栈和队列

栈 一、栈的概念及结构 栈&#xff08;Stack&#xff09;是一种特殊的线性表&#xff0c;其只允许在表的固定的一端进行插入和删除操作。 栈顶&#xff1a;进行插入数据和删除数据的一端。 栈底&#xff1a;相对于栈顶的另一端。 原则&#xff1a;栈的数据元素遵循后进先出…

【Python报错已解决】`TypeError`:`TypeError: not enough arguments for format string`

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 引言 在Python编程中&#xff0c;TypeError是一个常见的错误类型&#xff0c;它表示在操作或函数调用中使用了错误的类型。本文…

服务器死机/无故宕机排查思路/服务器起不来

1、查看服务器型号 dmidecode -t system dmidecode | grep ‘Product Name’ 2、风扇异响&#xff1a;查看BMC&#xff0c;坏了一个风扇其他的所有的风扇会全速转。 3、服务器亮红灯 红灯就是 故障告警 不一定是啥需要查看BMC口日志。这种就是看bmc日志 会有打印的 -问题现象&a…

java中的反射Reflection

Java中的反射&#xff08;Reflection&#xff09;是一种强大的机制&#xff0c;它允许程序在运行时查询和操作对象的类型信息。通过反射API&#xff0c;程序可以动态地创建对象、调用方法、访问字段和构造函数等&#xff0c;即使在编译时这些信息是未知的。 反射的原理 反射的…

微软分享其首款定制人工智能芯片Maia 100的更多细节

在2023年Ignite大会期间&#xff0c;微软首次宣布其已开发出名为Maia的自主人工智能加速器芯片。今年早些时候&#xff0c;在Build开发者大会上&#xff0c;微软分享了其首个自主研发的人工智能加速器Azure Maia 100的更多细节。Maia 100 是台积电 5nm 节点上制造的最大处理器之…

深拷贝与浅拷贝的区别

浅拷贝会导致深层数据改变&#xff0c;而深拷贝不会改变任何数据。 简单说就是&#xff1a; 浅拷贝只复制某个对象的引用&#xff0c;而不复制对象本身&#xff0c;新旧对象还是共享同一块内存。 深拷贝会创造一个一模一样的对象&#xff0c;新对象和原对象不共享内存&#x…

江协科技stm32————10-4 I2C通信协议

目录 I2C外设简介 I2C功能框图 基本结构图&#xff08;一主多从&#xff09; GPIO复用输入输出图 主机发送​编辑 START stop ​ EV5 &#xff08;标志位&#xff09; BTF 主机接收 ACK 软件/硬件波形对比 I2C外设简介 可变多主机模型11110作为10位地址模式的标志位…

「OC」初识MVC —— 简单学习UITableView的解耦

「OC」初识MVC —— 简单学习UITableView的解耦 文章目录 「OC」初识MVC —— 简单学习UITableView的解耦写在前面认识MVC解耦数据源代理 创建cell的基础类创建section的相关类分离数据源分离代理总结参考资料 写在前面 最近在学习了解MVC&#xff0c;然后开始发现&#xff0c…

VSCode连接SSH发生connection timeout

想用用实验室服务器连接vscode跑HM编码&#xff0c;今天突然连服务器连不到&#xff0c;报错&#xff1a;[13:47:13.068] Opening exec server for ssh-remotea406-server-lan [13:47:13.082] Initizing new exec server for ssh-remotea406-server-lan [13:47:13.089] Using c…

Apple LLM: 智能基础语言模型(AFM)

今天想和大家分享一下我最近在arXiv.org上看到苹果发表的一篇技术论文 Apple Intelligence Foundation Language Models (https://arxiv.org/abs/2407.21075)&#xff0c;概述了他们的模型训练。这虽然出乎意料&#xff0c;但绝对是一个积极的惊喜&#xff01; 这篇论文有那么多…

Dubbo如何传递链路追踪id?

1.什么是链路追踪&#xff1f; 分布式链路追踪就是将一次分布式请求还原成调用链路&#xff0c;将一次分布式请求的调用情况集中展示&#xff0c;比如各个服务节点上的耗时、请求具体到达哪台机器上、每个服务节点的请求状态等等。 链路跟踪主要功能&#xff1a; 故障快速定…

ET6框架(五)ECS组件式编程

文章目录 一、什么是ECS:二、ECS编程原则&#xff1a;三、组件生命周期&#xff1a; 一、什么是ECS: ECS即Entity、Component、System的简称&#xff0c;遵循组合优于继承的原则 Entity(实体) : 实体是一个概念&#xff0c;是指存在你游戏世界中的一个独立物体&#xff0c;是一…

图机器学习实战:从数据源到可视化

《动手学图机器学习》并不是一本纯粹介绍图机器学习理论的著作&#xff0c;Alessandro Negro 博士作为科学家和 Reco4 公司的 CEO&#xff0c;长期维护图数据源的推荐系统。他结合机器学习工程和图机器学习方法&#xff0c;通过推荐引擎、欺诈检测和知识图谱等案例&#xff0c;…

【书生2.4】InternLM + LlamaIndex RAG 实践

InternLM LlamaIndex RAG 实践 1 环境安装2 材料准备3 还原实验4 更换问题验证rag 【Intern Studio的gpu不足。本实验使用自有服务器】 1 环境安装 conda create -n llamaindex python3.10 conda activate llamaindex conda install pytorch2.0.1 torchvision0.15.2 torchau…

新160个crackme - 042-crackme

运行分析 提示需要key.dat文件 同目录下创建一个key.dat&#xff0c;提示key.dat内容不对 PE分析 C程序&#xff0c;32位&#xff0c;无壳 静态分析&动态调试 ida搜索字符串&#xff0c;找到关键字符串双击进入关键函数 对关键函数进行分析&#xff0c;注释如上当满足67行公…

信息安全数学基础(4)最大公因数

前言 在信息安全数学基础中&#xff0c;最大公因数&#xff08;Greatest Common Divisor, GCD&#xff09;是一个核心概念&#xff0c;它在密码学、数论等多个领域都有广泛应用。以下是对最大公因数的详细阐述&#xff1a; 一、定义 设a和b是两个非零整数&#xff0c;若整数d同…