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

2024-08-31 06:28

本文主要是介绍springboot+redis+mybatis体会布隆过滤器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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 {@Autowiredprivate BloomService bloomService;@RequestMapping(value = "/addUser")@ResponseBodypublic 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")@ResponseBodypublic 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:";@Resourceprivate BloomDao bloomDao;@Autowiredprivate RedisTemplate redisTemplate;@Autowiredprivate BloomCheckUtil bloomCheckUtil;@Overridepublic void addUser(User user) {int i = bloomDao.addUser(user);if(i>0){//mysql插入成功,将数据存入redisUser currentUser = bloomDao.getUser(user.getId());redisTemplate.opsForValue().set(redisKey+user.getId(), JSONObject.toJSONString(currentUser));}}@Overridepublic 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;}@Overridepublic 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";@Autowiredprivate RedisTemplate redisTemplate;@Autowiredprivate BloomService bloomService;@PostConstructpublic 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的值更新成1redisTemplate.opsForValue().setBit(key,index,true);}}
}
@Component
public class BloomCheckUtil {private static final String key = "LEGITIMATE_USER";@Autowiredprivate 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
在这里插入图片描述

这篇关于springboot+redis+mybatis体会布隆过滤器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1123137

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

mybatis的整体架构

mybatis的整体架构分为三层: 1.基础支持层 该层包括:数据源模块、事务管理模块、缓存模块、Binding模块、反射模块、类型转换模块、日志模块、资源加载模块、解析器模块 2.核心处理层 该层包括:配置解析、参数映射、SQL解析、SQL执行、结果集映射、插件 3.接口层 该层包括:SqlSession 基础支持层 该层保护mybatis的基础模块,它们为核心处理层提供了良好的支撑。

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听