⭐⭐ Redis专栏:Redis专栏
⭐⭐ 个人主页:个人主页
目录
一.SSM整合Redis
1.2配制Spring-redis.xml文件
1.3 applicationContext.xml
1.4 配置redis的key生成策略
二.redis的注解式开发
2.1 redis的注解式是什么及其好处
2.2 redis注解式的场景应用
①Cacheable
②@CachePut
③@CacheEvict
2.3 Cacheable 与CachePut的区别
三、Redis中的击穿、穿透、雪崩的三种场景
3.1 击穿
3.2. 穿透
3.3. 雪崩
1.1 配置文件pom.xml文件
添加redis依赖:
2.9.0 1.7.1.RELEASE redis.clients jedis ${redis.version} org.springframework.data spring-data-redis ${redis.spring.version}
并且为了解决mybatis-generator-maven-plugin运行时没有将jdbc.properites文件放入target文件夹的问题
src/main/resources *.properties *.xml
4.0.0
org.example
ssm2
1.0-SNAPSHOT
war
ssm2 Maven Webapp
http://www.example.com
UTF-8
1.8
1.8
3.7.0
5.0.2.RELEASE
3.4.5
5.1.44
5.1.2
1.3.1
2.1.1
2.4.3
2.9.1
4.12
4.0.0
1.18.2
2.10.0
1.7.7
2.9.0
1.7.1.RELEASE
org.springframework
spring-context
${spring.version}
org.springframework
spring-orm
${spring.version}
org.springframework
spring-tx
${spring.version}
org.springframework
spring-aspects
${spring.version}
org.springframework
spring-web
${spring.version}
org.springframework
spring-test
${spring.version}
org.mybatis
mybatis
${mybatis.version}
mysql
mysql-connector-java
${mysql.version}
com.github.pagehelper
pagehelper
${pagehelper.version}
org.mybatis
mybatis-spring
${mybatis.spring.version}
org.apache.commons
commons-dbcp2
${commons.dbcp2.version}
org.apache.commons
commons-pool2
${commons.pool2.version}
org.apache.logging.log4j
log4j-core
${log4j2.version}
org.apache.logging.log4j
log4j-api
${log4j2.version}
org.apache.logging.log4j
log4j-web
${log4j2.version}
junit
junit
${junit.version}
test
javax.servlet
javax.servlet-api
${servlet.version}
provided
org.projectlombok
lombok
${lombok.version}
provided
org.springframework
spring-webmvc
${spring.version}
javax.servlet.jsp
javax.servlet.jsp-api
2.3.3
jstl
jstl
1.2
taglibs
standard
1.1.2
commons-fileupload
commons-fileupload
1.3.3
org.hibernate
hibernate-validator
6.0.7.Final
com.fasterxml.jackson.core
jackson-databind
2.9.3
com.fasterxml.jackson.core
jackson-core
2.9.3
com.fasterxml.jackson.core
jackson-annotations
2.9.3
org.apache.shiro
shiro-core
1.3.2
org.apache.shiro
shiro-web
1.3.2
org.apache.shiro
shiro-spring
1.3.2
net.sf.ehcache
ehcache
${ehcache.version}
org.slf4j
slf4j-api
${slf4j-api.version}
org.slf4j
jcl-over-slf4j
${slf4j-api.version}
runtime
org.apache.logging.log4j
log4j-slf4j-impl
${log4j2.version}
redis.clients
jedis
${redis.version}
org.springframework.data
spring-data-redis
${redis.spring.version}
ssm2
src/main/java
**/*.xml
src/main/resources
*.properties
*.xml
org.apache.maven.plugins
maven-compiler-plugin
${maven.compiler.plugin.version}
${maven.compiler.target}
${project.build.sourceEncoding}
org.mybatis.generator
mybatis-generator-maven-plugin
1.3.2
mysql
mysql-connector-java
${mysql.version}
true
maven-clean-plugin
3.1.0
maven-resources-plugin
3.0.2
maven-compiler-plugin
3.8.0
maven-surefire-plugin
2.22.1
maven-war-plugin
3.2.2
maven-install-plugin
2.5.2
maven-deploy-plugin
2.8.2
在这个里面就是redis连接池以及工厂的配置
redis.hostName=47.100.191.44
redis.port=6379
redis.password=xiaoli_redis
redis.timeout=10000
redis.maxIdle=300
redis.maxTotal=1000
redis.maxWaitMillis=1000
redis.minEvictableIdleTimeMillis=300000
redis.numTestsPerEvictionRun=1024
redis.timeBetweenEvictionRunsMillis=30000
redis.testOnBorrow=true
redis.testWhileIdle=true
redis.expiration=3600
resouces的配置必须要涵盖读取.properties结尾的文件
classpath:jdbc.properties
classpath:redis.properties
package com.zking.ssm.redis;
import lombok.extern.slf4j.Slf4j;
import org.springframework.cache.interceptor.KeyGenerator;
import org.springframework.util.ClassUtils;
import java.lang.reflect.Array;
import java.lang.reflect.Method;
@Slf4j
public class CacheKeyGenerator implements KeyGenerator {
// custom cache key
public static final int NO_PARAM_KEY = 0;
public static final int NULL_PARAM_KEY = 53;
@Override
public Object generate(Object target, Method method, Object... params) {
StringBuilder key = new StringBuilder();
//键的定义规则
key.append(target.getClass().getSimpleName()).append(".").append(method.getName()).append(":");
if (params.length == 0) {
key.append(NO_PARAM_KEY);
} else {
int count = 0;
for (Object param : params) {
if (0 != count) {//参数之间用,进行分隔
key.append(',');
}
if (param == null) {
key.append(NULL_PARAM_KEY);
} else if (ClassUtils.isPrimitiveArray(param.getClass())) {
int length = Array.getLength(param);
for (int i = 0; i < length; i++) {
key.append(Array.get(param, i));
key.append(',');
}
} else if (ClassUtils.isPrimitiveOrWrapper(param.getClass()) || param instanceof String) {
key.append(param);
} else {//Java一定要重写hashCode和eqauls
key.append(param.hashCode());
}
count++;
}
}
String finalKey = key.toString();
// IEDA要安装lombok插件
log.debug("using cache key={}", finalKey);
return finalKey;
}
}
Redis的注解式是指通过注解的方式来简化对Redis的操作。在使用注解式的方法中,程序员可以通过在方法上添加特定的注解来实现对Redis的读写操作,而不需要显式地编写Redis相关的代码。使用注解式操作Redis可以简化代码编写,提高开发效率。同时,注解式操作还可以利用Spring的AOP特性,实现缓存、事务管理等功能,进一步提升应用的性能和可维护性。通过注解式,可以方便地对Redis的缓存策略进行配置。可以使用注解来控制缓存的过期时间、缓存刷新策略等,实现更加灵活和高效的缓存管理。
通过在方法上添加@RedisHash、@RedisId、@RedisIndexed等注解,可以将Java对象映射到Redis中的Hash数据结构,并实现对Redis的CRUD操作。
同时,Spring Data Redis还提供了一些常用的注解,如@RedisHashIndexed、@RedisHashKey、@RedisHashReference等,用于实现索引、关联等功能。
建立一个测试类:
package com.zking.shiro;
import com.zking.ssm.biz.ClazzBiz;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
/**
* @author姿势不太差
* @site www.javayz.com
* @company xxx公司
* @create 2022-10-26 15:29
*/
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"classpath:applicationContext.xml"})
public class ClazzBizTest {
@Autowired
private ClazzBiz clazzBiz;
@Test
public void test1(){
System.out.println(clazzBiz.selectByPrimaryKey(10));
System.out.println(clazzBiz.selectByPrimaryKey(10));
}
@Test
public void test2(){
clazzBiz.deleteByPrimaryKey(10);
}
}
在class.Biz里面:
@Cacheable("clz")
Clazz selectByPrimaryKey(Integer cid);
在测试类里面:
@Test
public void test1(){
System.out.println(clazzBiz.selectByPrimaryKey(1));
System.out.println(clazzBiz.selectByPrimaryKey(1));
}
运行结果,只会走一次数据库,之后都是走redis缓存,减少了数据库压力
@CachePut(value = "xx",key = "'cid:'+#cid")
Clazz selectByPrimaryKey(Integer cid);
@Test
public void test1(){
System.out.println(clazzBiz.selectByPrimaryKey(1));
System.out.println(clazzBiz.selectByPrimaryKey(1));
}
两次都会走数据库:
用于清除缓存的数据。它通常用于方法上,表示在方法执行后清除指定的缓存。
@CacheEvict(value = "xx",key = "'cid:'+#cid",allEntries = true)
int deleteByPrimaryKey(Integer cid);
cacheable会在redis中存储数据,同时也会读取数据;cacheput只会在redis中数据,不会读数据
1. @Cacheable注解用于读取缓存,如果缓存中存在对应的结果则直接返回,不执行方法体内的代码;@CachePut注解则用于更新缓存,无论缓存中是否存在对应的结果都会执行方法体内的代码。
2. @Cacheable注解可以指定缓存名称和缓存键,用于唯一标识缓存的存储位置;@CachePut注解不需要指定缓存键,它会根据方法的参数自动生成缓存键。
3. @Cacheable注解适合用于查询操作,可以减少数据库的访问压力;@CachePut注解适合用于更新操作,可以保证缓存的数据与数据库的数据保持一致。
含义:击穿指的是当一个缓存键在缓存中不存在,且同时有大量请求同时访问这个缓存键时,这些请求会直接访问后端数据库,导致数据库压力骤增。这种情况通常发生在热点数据失效导致的缓存穿透问题。
场景:某个热点数据在缓存中过期或被手动删除,然后有大量的请求同时访问这个数据。
解决方案:一种解决方案是在缓存失效时使用互斥锁,只允许一个请求去更新缓存,其他请求等待并读取更新后的缓存。另一种方案是使用“空值缓存”,即将缓存中不存在的数据(如数据库中不存在的数据)也缓存起来,这样下次请求同样的数据时,可以直接从缓存中获取到空值。
含义: 穿透指的是当一个请求查询一个不存在的缓存键时,由于缓存中没有这个键的值,请求会直接访问后端数据库,同样导致数据库压力增加。这种情况通常发生在恶意攻击或异常查询导致的缓存穿透问题。
场景:某个恶意用户或者异常请求不断查询缓存中不存在的数据。
解决方案:一种解决方案是使用布隆过滤器(Bloom Filter),在查询缓存之前先进行布隆过滤器的判断,如果键不在布隆过滤器中,则直接返回缓存缺失的结果,而不去查询数据库。另一种方案是在缓存中设置一个过期时间,即使不存在的数据也缓存一段时间,避免频繁查询数据库。
含义:雪崩指的是当缓存中的大量数据同时失效或者由于某种原因都不可用时,所有的请求都直接访问后端数据库,导致数据库压力骤增。这种情况通常发生在缓存服务器故障、断电、网络异常等情况下。
场景:缓存服务器故障、网络异常等导致缓存中的大量数据同时失效。
解决方案:一种解决方案是在缓存的失效时间上引入随机性,避免大量缓存同时失效。另一种方案是使用多级缓存架构,将缓存分为多个层级,当某一层缓存失效时,可以从其他层级的缓存中获取数据,避免全部请求直接访问数据库。