SSM之Spring注解式缓存Redis以及redies中的击穿,雪崩,穿的三种现象

SSM之Spring注解式缓存Redis以及redies中的击穿,雪崩,穿的三种现象_第1张图片

                                                ⭐⭐ 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. 雪崩


 一.SSM整合Redis

        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.source}
            ${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
        
      
    
  

        1.2配制Spring-redis.xml文件

                 在这个里面就是redis连接池以及工厂的配置

   redis.properties.xml

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

 Spring-redis.xml




    
    

    
    
        
        
        
        
        
        
        
        
        
        
        
        
        
        
        
        
    

    
    
        
        
        
        
        
        
        
        
        
    

    
    
        
        
        
            
        
        
            
        
        
            
        
        
            
        
        
        
    

    
    
        
        
        
        
        
        
        
            
                
            
        
    

    
    

    
    

1.3 applicationContext.xml

        resouces的配置必须要涵盖读取.properties结尾的文件



    
    
        
        
        
            
                classpath:jdbc.properties
                classpath:redis.properties
            
        
    


    
    
    

1.4 配置redis的key生成策略

        CacheKeyGenerator.java
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的注解式开发

              2.1  redis的注解式是什么及其好处

        Redis的注解式是指通过注解的方式来简化对Redis的操作。在使用注解式的方法中,程序员可以通过在方法上添加特定的注解来实现对Redis的读写操作,而不需要显式地编写Redis相关的代码。使用注解式操作Redis可以简化代码编写,提高开发效率。同时,注解式操作还可以利用Spring的AOP特性,实现缓存、事务管理等功能,进一步提升应用的性能和可维护性。通过注解式,可以方便地对Redis的缓存策略进行配置。可以使用注解来控制缓存的过期时间、缓存刷新策略等,实现更加灵活和高效的缓存管理。

        通过在方法上添加@RedisHash、@RedisId、@RedisIndexed等注解,可以将Java对象映射到Redis中的Hash数据结构,并实现对Redis的CRUD操作。

        同时,Spring Data Redis还提供了一些常用的注解,如@RedisHashIndexed、@RedisHashKey、@RedisHashReference等,用于实现索引、关联等功能。

        2.2 redis注解式的场景应用

                建立一个测试类:

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);
    }
}

①Cacheable                                                                                       

       在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缓存,减少了数据库压力

SSM之Spring注解式缓存Redis以及redies中的击穿,雪崩,穿的三种现象_第2张图片SSM之Spring注解式缓存Redis以及redies中的击穿,雪崩,穿的三种现象_第3张图片

 ②@CachePut

 @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));
    }

两次都会走数据库:

SSM之Spring注解式缓存Redis以及redies中的击穿,雪崩,穿的三种现象_第4张图片

 ③@CacheEvict

用于清除缓存的数据。它通常用于方法上,表示在方法执行后清除指定的缓存。

 @CacheEvict(value = "xx",key = "'cid:'+#cid",allEntries = true)
    int deleteByPrimaryKey(Integer cid);

2.3 Cacheable 与CachePut的区别

        cacheable会在redis中存储数据,同时也会读取数据;cacheput只会在redis中数据,不会读数据

        1. @Cacheable注解用于读取缓存,如果缓存中存在对应的结果则直接返回,不执行方法体内的代码;@CachePut注解则用于更新缓存,无论缓存中是否存在对应的结果都会执行方法体内的代码。

        2. @Cacheable注解可以指定缓存名称和缓存键,用于唯一标识缓存的存储位置;@CachePut注解不需要指定缓存键,它会根据方法的参数自动生成缓存键。

        3. @Cacheable注解适合用于查询操作,可以减少数据库的访问压力;@CachePut注解适合用于更新操作,可以保证缓存的数据与数据库的数据保持一致。

    三、Redis中的击穿、穿透、雪崩的三种场景

SSM之Spring注解式缓存Redis以及redies中的击穿,雪崩,穿的三种现象_第5张图片

3.1 击穿

     含义:击穿指的是当一个缓存键在缓存中不存在,且同时有大量请求同时访问这个缓存键时,这些请求会直接访问后端数据库,导致数据库压力骤增。这种情况通常发生在热点数据失效导致的缓存穿透问题。
     场景:某个热点数据在缓存中过期或被手动删除,然后有大量的请求同时访问这个数据。

解决方案:一种解决方案是在缓存失效时使用互斥锁,只允许一个请求去更新缓存,其他请求等待并读取更新后的缓存。另一种方案是使用“空值缓存”,即将缓存中不存在的数据(如数据库中不存在的数据)也缓存起来,这样下次请求同样的数据时,可以直接从缓存中获取到空值。

3.2. 穿透

    含义: 穿透指的是当一个请求查询一个不存在的缓存键时,由于缓存中没有这个键的值,请求会直接访问后端数据库,同样导致数据库压力增加。这种情况通常发生在恶意攻击或异常查询导致的缓存穿透问题。

场景:某个恶意用户或者异常请求不断查询缓存中不存在的数据。

解决方案:一种解决方案是使用布隆过滤器(Bloom Filter),在查询缓存之前先进行布隆过滤器的判断,如果键不在布隆过滤器中,则直接返回缓存缺失的结果,而不去查询数据库。另一种方案是在缓存中设置一个过期时间,即使不存在的数据也缓存一段时间,避免频繁查询数据库。

3.3. 雪崩

        含义:雪崩指的是当缓存中的大量数据同时失效或者由于某种原因都不可用时,所有的请求都直接访问后端数据库,导致数据库压力骤增。这种情况通常发生在缓存服务器故障、断电、网络异常等情况下。

场景:缓存服务器故障、网络异常等导致缓存中的大量数据同时失效。

解决方案:一种解决方案是在缓存的失效时间上引入随机性,避免大量缓存同时失效。另一种方案是使用多级缓存架构,将缓存分为多个层级,当某一层缓存失效时,可以从其他层级的缓存中获取数据,避免全部请求直接访问数据库。

今天的分享就到这啦!!

你可能感兴趣的:(Redis的入门级教程,缓存,spring,redis,运维,服务器,linux,java)