Redis は Spring アノテーション キャッシュを SSM と統合します

目次

1. 統合

1.1. 統合されたアプリケーション

1.1.1.pom設定

1.1.2.必要な構成

2. アノテーションの開発と適用シナリオ

2.1. @キャッシュ可能

2.2. @CachePut 

2.3. @CacheEvict

2.4. 概要

3. Redis の破壊と侵入雪崩

                今日はそれです! !お役に立てれば! !​ 


1. 統合

1.1. 統合されたアプリケーション

1.1.1.pom設定

プロジェクトの pom.xml ファイルに Redis 依存関係を追加します。

 以下はすべてインポートされた依存関係です。 

<?xml version="1.0" encoding="UTF-8"?>
 
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  <modelVersion>4.0.0</modelVersion>
 
  <groupId>org.example</groupId>
  <artifactId>ssm2</artifactId>
  <version>1.0-SNAPSHOT</version>
  <packaging>war</packaging>
 
  <name>ssm2 Maven Webapp</name>
  <!-- FIXME change it to the project's website -->
  <url>http://www.example.com</url>
 
  <properties>
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    <maven.compiler.source>1.8</maven.compiler.source>
    <maven.compiler.target>1.8</maven.compiler.target>
    <maven.compiler.plugin.version>3.7.0</maven.compiler.plugin.version>
 
    <!--添加jar包依赖-->
    <!--1.spring 5.0.2.RELEASE相关-->
    <spring.version>5.0.2.RELEASE</spring.version>
    <!--2.mybatis相关-->
    <mybatis.version>3.4.5</mybatis.version>
    <!--mysql-->
    <mysql.version>5.1.44</mysql.version>
    <!--pagehelper分页jar依赖-->
    <pagehelper.version>5.1.2</pagehelper.version>
    <!--mybatis与spring集成jar依赖-->
    <mybatis.spring.version>1.3.1</mybatis.spring.version>
    <!--3.dbcp2连接池相关 druid-->
    <commons.dbcp2.version>2.1.1</commons.dbcp2.version>
    <commons.pool2.version>2.4.3</commons.pool2.version>
    <!--4.log日志相关-->
    <log4j2.version>2.9.1</log4j2.version>
    <!--5.其他-->
    <junit.version>4.12</junit.version>
    <servlet.version>4.0.0</servlet.version>
    <lombok.version>1.18.2</lombok.version>
 
    <ehcache.version>2.10.0</ehcache.version>
    <slf4j-api.version>1.7.7</slf4j-api.version>
 
    <redis.version>2.9.0</redis.version>
    <redis.spring.version>1.7.1.RELEASE</redis.spring.version>
  </properties>
 
  <dependencies>
    <!--1.spring相关-->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context</artifactId>
      <version>${spring.version}</version>
    </dependency>
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-orm</artifactId>
      <version>${spring.version}</version>
    </dependency>
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-tx</artifactId>
      <version>${spring.version}</version>
    </dependency>
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-aspects</artifactId>
      <version>${spring.version}</version>
    </dependency>
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-web</artifactId>
      <version>${spring.version}</version>
    </dependency>
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-test</artifactId>
      <version>${spring.version}</version>
    </dependency>
 
    <!--2.mybatis相关-->
    <dependency>
      <groupId>org.mybatis</groupId>
      <artifactId>mybatis</artifactId>
      <version>${mybatis.version}</version>
    </dependency>
    <!--mysql-->
    <dependency>
      <groupId>mysql</groupId>
      <artifactId>mysql-connector-java</artifactId>
      <version>${mysql.version}</version>
    </dependency>
    <!--pagehelper分页插件jar包依赖-->
    <dependency>
      <groupId>com.github.pagehelper</groupId>
      <artifactId>pagehelper</artifactId>
      <version>${pagehelper.version}</version>
    </dependency>
    <!--mybatis与spring集成jar包依赖-->
    <dependency>
      <groupId>org.mybatis</groupId>
      <artifactId>mybatis-spring</artifactId>
      <version>${mybatis.spring.version}</version>
    </dependency>
 
    <!--3.dbcp2连接池相关-->
    <dependency>
      <groupId>org.apache.commons</groupId>
      <artifactId>commons-dbcp2</artifactId>
      <version>${commons.dbcp2.version}</version>
    </dependency>
    <dependency>
      <groupId>org.apache.commons</groupId>
      <artifactId>commons-pool2</artifactId>
      <version>${commons.pool2.version}</version>
    </dependency>
 
    <!--4.log日志相关依赖-->
    <!--核心log4j2jar包-->
    <dependency>
      <groupId>org.apache.logging.log4j</groupId>
      <artifactId>log4j-core</artifactId>
      <version>${log4j2.version}</version>
    </dependency>
    <dependency>
      <groupId>org.apache.logging.log4j</groupId>
      <artifactId>log4j-api</artifactId>
      <version>${log4j2.version}</version>
    </dependency>
    <!--web工程需要包含log4j-web,非web工程不需要-->
    <dependency>
      <groupId>org.apache.logging.log4j</groupId>
      <artifactId>log4j-web</artifactId>
      <version>${log4j2.version}</version>
    </dependency>
 
    <!--5.其他-->
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>${junit.version}</version>
      <scope>test</scope>
    </dependency>
    <dependency>
      <groupId>javax.servlet</groupId>
      <artifactId>javax.servlet-api</artifactId>
      <version>${servlet.version}</version>
      <scope>provided</scope>
    </dependency>
    <dependency>
      <groupId>org.projectlombok</groupId>
      <artifactId>lombok</artifactId>
      <version>${lombok.version}</version>
      <scope>provided</scope>
    </dependency>
 
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-webmvc</artifactId>
      <version>${spring.version}</version>
    </dependency>
 
    <!-- jsp依赖-->
    <dependency>
      <groupId>javax.servlet.jsp</groupId>
      <artifactId>javax.servlet.jsp-api</artifactId>
      <version>2.3.3</version>
    </dependency>
    <dependency>
      <groupId>jstl</groupId>
      <artifactId>jstl</artifactId>
      <version>1.2</version>
    </dependency>
    <dependency>
      <groupId>taglibs</groupId>
      <artifactId>standard</artifactId>
      <version>1.1.2</version>
    </dependency>
 
    <dependency>
      <groupId>commons-fileupload</groupId>
      <artifactId>commons-fileupload</artifactId>
      <version>1.3.3</version>
    </dependency>
 
<!--    做服务端参数校验 JSR303 的jar包依赖 -->
    <dependency>
      <groupId>org.hibernate</groupId>
      <artifactId>hibernate-validator</artifactId>
      <version>6.0.7.Final</version>
    </dependency>
 
<!--    用来SpringMVC支持json数据转换-->
    <dependency>
      <groupId>com.fasterxml.jackson.core</groupId>
      <artifactId>jackson-databind</artifactId>
      <version>2.9.3</version>
    </dependency>
    <dependency>
      <groupId>com.fasterxml.jackson.core</groupId>
      <artifactId>jackson-core</artifactId>
      <version>2.9.3</version>
    </dependency>
    <dependency>
      <groupId>com.fasterxml.jackson.core</groupId>
      <artifactId>jackson-annotations</artifactId>
      <version>2.9.3</version>
    </dependency>
 
<!--    shiro相关依赖 -->
    <dependency>
      <groupId>org.apache.shiro</groupId>
      <artifactId>shiro-core</artifactId>
      <version>1.3.2</version>
    </dependency>
 
    <dependency>
      <groupId>org.apache.shiro</groupId>
      <artifactId>shiro-web</artifactId>
      <version>1.3.2</version>
    </dependency>
 
    <dependency>
      <groupId>org.apache.shiro</groupId>
      <artifactId>shiro-spring</artifactId>
      <version>1.3.2</version>
    </dependency>
 
    <dependency>
      <groupId>net.sf.ehcache</groupId>
      <artifactId>ehcache</artifactId>
      <version>${ehcache.version}</version>
    </dependency>
 
    <!-- slf4j核心包 -->
    <dependency>
      <groupId>org.slf4j</groupId>
      <artifactId>slf4j-api</artifactId>
      <version>${slf4j-api.version}</version>
    </dependency>
    <dependency>
      <groupId>org.slf4j</groupId>
      <artifactId>jcl-over-slf4j</artifactId>
      <version>${slf4j-api.version}</version>
      <scope>runtime</scope>
    </dependency>
 
    <!--用于与slf4j保持桥接 -->
    <dependency>
      <groupId>org.apache.logging.log4j</groupId>
      <artifactId>log4j-slf4j-impl</artifactId>
      <version>${log4j2.version}</version>
    </dependency>
 
    <dependency>
      <groupId>redis.clients</groupId>
      <artifactId>jedis</artifactId>
      <version>${redis.version}</version>
    </dependency>
    <dependency>
      <groupId>org.springframework.data</groupId>
      <artifactId>spring-data-redis</artifactId>
      <version>${redis.spring.version}</version>
    </dependency>
  </dependencies>
 
  <build>
    <finalName>ssm2</finalName>
    <resources>
      <!--解决mybatis-generator-maven-plugin运行时没有将XxxMapper.xml文件放入target文件夹的问题-->
      <resource>
        <directory>src/main/java</directory>
        <includes>
          <include>**/*.xml</include>
        </includes>
      </resource>
      <!--解决mybatis-generator-maven-plugin运行时没有将jdbc.properites文件放入target文件夹的问题-->
      <resource>
        <directory>src/main/resources</directory>
        <includes>
          <include>*.properties</include>
          <include>*.xml</include>
        </includes>
      </resource>
    </resources>
    <pluginManagement><!-- lock down plugins versions to avoid using Maven defaults (may be moved to parent pom) -->
      <plugins>
        <plugin>
          <groupId>org.apache.maven.plugins</groupId>
          <artifactId>maven-compiler-plugin</artifactId>
          <version>${maven.compiler.plugin.version}</version>
          <configuration>
            <source>${maven.compiler.source}</source>
            <target>${maven.compiler.target}</target>
            <encoding>${project.build.sourceEncoding}</encoding>
          </configuration>
        </plugin>
        <plugin>
          <groupId>org.mybatis.generator</groupId>
          <artifactId>mybatis-generator-maven-plugin</artifactId>
          <version>1.3.2</version>
          <dependencies>
            <!--使用Mybatis-generator插件不能使用太高版本的mysql驱动 -->
            <dependency>
              <groupId>mysql</groupId>
              <artifactId>mysql-connector-java</artifactId>
              <version>${mysql.version}</version>
            </dependency>
          </dependencies>
          <configuration>
            <overwrite>true</overwrite>
          </configuration>
        </plugin>
 
        <plugin>
          <artifactId>maven-clean-plugin</artifactId>
          <version>3.1.0</version>
        </plugin>
        <!-- see http://maven.apache.org/ref/current/maven-core/default-bindings.html#Plugin_bindings_for_war_packaging -->
        <plugin>
          <artifactId>maven-resources-plugin</artifactId>
          <version>3.0.2</version>
        </plugin>
        <plugin>
          <artifactId>maven-compiler-plugin</artifactId>
          <version>3.8.0</version>
        </plugin>
        <plugin>
          <artifactId>maven-surefire-plugin</artifactId>
          <version>2.22.1</version>
        </plugin>
        <plugin>
          <artifactId>maven-war-plugin</artifactId>
          <version>3.2.2</version>
        </plugin>
        <plugin>
          <artifactId>maven-install-plugin</artifactId>
          <version>2.5.2</version>
        </plugin>
        <plugin>
          <artifactId>maven-deploy-plugin</artifactId>
          <version>2.8.2</version>
        </plugin>
      </plugins>
    </pluginManagement>
  </build>
</project>

1.1.2.必要な構成

SSM プロジェクトで spring-readis.xml を作成し、Redis 接続情報を構成し、データ ソースを構成し、シリアライザーと Redis キー生成を構成します。戦略、RedisTemplate を構成します。

作成 redis.properties ホスト名、ポート番号、パスワードなどのデータ接続情報を書き込みます。

redis.hostName=localhost
redis.port=6379
redis.password=123456
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 構成ファイルを作成し、その中で以下を構成します

Redis接続プールの構成

 <bean id="poolConfig" class="redis.clients.jedis.JedisPoolConfig">
        <!--最大空闲数-->
        <property name="maxIdle" value="${redis.maxIdle}"/>
        <!--连接池的最大数据库连接数  -->
        <property name="maxTotal" value="${redis.maxTotal}"/>
        <!--最大建立连接等待时间-->
        <property name="maxWaitMillis" value="${redis.maxWaitMillis}"/>
        <!--逐出连接的最小空闲时间 默认1800000毫秒(30分钟)-->
        <property name="minEvictableIdleTimeMillis" value="${redis.minEvictableIdleTimeMillis}"/>
        <!--每次逐出检查时 逐出的最大数目 如果为负数就是 : 1/abs(n), 默认3-->
        <property name="numTestsPerEvictionRun" value="${redis.numTestsPerEvictionRun}"/>
        <!--逐出扫描的时间间隔(毫秒) 如果为负数,则不运行逐出线程, 默认-1-->
        <property name="timeBetweenEvictionRunsMillis" value="${redis.timeBetweenEvictionRunsMillis}"/>
        <!--是否在从池中取出连接前进行检验,如果检验失败,则从池中去除连接并尝试取出另一个-->
        <property name="testOnBorrow" value="${redis.testOnBorrow}"/>
        <!--在空闲时检查有效性, 默认false  -->
        <property name="testWhileIdle" value="${redis.testWhileIdle}"/>
    </bean>

Redis 接続ファクトリーを構成する

    <!-- 3. redis连接工厂 -->
    <bean id="connectionFactory" class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory"
          destroy-method="destroy">
        <property name="poolConfig" ref="poolConfig"/>
        <!--IP地址 -->
        <property name="hostName" value="${redis.hostName}"/>
        <!--端口号  -->
        <property name="port" value="${redis.port}"/>
        <!--如果Redis设置有密码  -->
        <property name="password" value="${redis.password}"/>
        <!--客户端超时时间单位是毫秒  -->
        <property name="timeout" value="${redis.timeout}"/>
    </bean>

Redis操作テンプレート

    <!-- 4. redis操作模板,使用该对象可以操作redis
        相当于session,专门操作数据库。
    -->
    <bean id="redisTemplate" class="org.springframework.data.redis.core.RedisTemplate">
        <property name="connectionFactory" ref="connectionFactory"/>
        <!--如果不配置Serializer,那么存储的时候缺省使用String,如果用User类型存储,那么会提示错误User can't cast to String!!  -->
        <property name="keySerializer">
            <bean class="org.springframework.data.redis.serializer.StringRedisSerializer"/>
        </property>
        <property name="valueSerializer">
            <bean class="org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer"/>
        </property>
        <property name="hashKeySerializer">
            <bean class="org.springframework.data.redis.serializer.StringRedisSerializer"/>
        </property>
        <property name="hashValueSerializer">
            <bean class="org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer"/>
        </property>
        <!--开启事务  -->
        <property name="enableTransactionSupport" value="true"/>
    </bean>

キャッシュマネージャーを構成する

    <!--  5.配置缓存管理器  -->
    <bean id="redisCacheManager" class="org.springframework.data.redis.cache.RedisCacheManager">
        <constructor-arg name="redisOperations" ref="redisTemplate"/>
        <!--redis缓存数据过期时间单位秒-->
        <property name="defaultExpiration" value="${redis.expiration}"/>
        <!--是否使用缓存前缀,与cachePrefix相关-->
        <property name="usePrefix" value="true"/>
        <!--配置缓存前缀名称-->
        <property name="cachePrefix">
            <bean class="org.springframework.data.redis.cache.DefaultRedisCachePrefix">
                <constructor-arg index="0" value="-cache-"/>
            </bean>
        </property>
    </bean>

作成 CacheKeyGenerator.java キャッシュ生成キー名の生成ルールを構成する

package com.junlinyi.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;
    }
}

 最後に spring-redis.xml 構成ファイルのすべての構成は次のとおりです。 

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:cache="http://www.springframework.org/schema/cache"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/context
       http://www.springframework.org/schema/context/spring-context.xsd
       http://www.springframework.org/schema/cache
       http://www.springframework.org/schema/cache/spring-cache.xsd">
 
    <!-- 1. 引入properties配置文件 -->
    <!--<context:property-placeholder location="classpath:redis.properties" />-->
 
    <!-- 2. redis连接池配置-->
    <bean id="poolConfig" class="redis.clients.jedis.JedisPoolConfig">
        <!--最大空闲数-->
        <property name="maxIdle" value="${redis.maxIdle}"/>
        <!--连接池的最大数据库连接数  -->
        <property name="maxTotal" value="${redis.maxTotal}"/>
        <!--最大建立连接等待时间-->
        <property name="maxWaitMillis" value="${redis.maxWaitMillis}"/>
        <!--逐出连接的最小空闲时间 默认1800000毫秒(30分钟)-->
        <property name="minEvictableIdleTimeMillis" value="${redis.minEvictableIdleTimeMillis}"/>
        <!--每次逐出检查时 逐出的最大数目 如果为负数就是 : 1/abs(n), 默认3-->
        <property name="numTestsPerEvictionRun" value="${redis.numTestsPerEvictionRun}"/>
        <!--逐出扫描的时间间隔(毫秒) 如果为负数,则不运行逐出线程, 默认-1-->
        <property name="timeBetweenEvictionRunsMillis" value="${redis.timeBetweenEvictionRunsMillis}"/>
        <!--是否在从池中取出连接前进行检验,如果检验失败,则从池中去除连接并尝试取出另一个-->
        <property name="testOnBorrow" value="${redis.testOnBorrow}"/>
        <!--在空闲时检查有效性, 默认false  -->
        <property name="testWhileIdle" value="${redis.testWhileIdle}"/>
    </bean>
 
    <!-- 3. redis连接工厂 -->
    <bean id="connectionFactory" class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory"
          destroy-method="destroy">
        <property name="poolConfig" ref="poolConfig"/>
        <!--IP地址 -->
        <property name="hostName" value="${redis.hostName}"/>
        <!--端口号  -->
        <property name="port" value="${redis.port}"/>
        <!--如果Redis设置有密码  -->
        <property name="password" value="${redis.password}"/>
        <!--客户端超时时间单位是毫秒  -->
        <property name="timeout" value="${redis.timeout}"/>
    </bean>
 
    <!-- 4. redis操作模板,使用该对象可以操作redis
        相当于session,专门操作数据库。
    -->
    <bean id="redisTemplate" class="org.springframework.data.redis.core.RedisTemplate">
        <property name="connectionFactory" ref="connectionFactory"/>
        <!--如果不配置Serializer,那么存储的时候缺省使用String,如果用User类型存储,那么会提示错误User can't cast to String!!  -->
        <property name="keySerializer">
            <bean class="org.springframework.data.redis.serializer.StringRedisSerializer"/>
        </property>
        <property name="valueSerializer">
            <bean class="org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer"/>
        </property>
        <property name="hashKeySerializer">
            <bean class="org.springframework.data.redis.serializer.StringRedisSerializer"/>
        </property>
        <property name="hashValueSerializer">
            <bean class="org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer"/>
        </property>
        <!--开启事务  -->
        <property name="enableTransactionSupport" value="true"/>
    </bean>
 
    <!--  5.配置缓存管理器  -->
    <bean id="redisCacheManager" class="org.springframework.data.redis.cache.RedisCacheManager">
        <constructor-arg name="redisOperations" ref="redisTemplate"/>
        <!--redis缓存数据过期时间单位秒-->
        <property name="defaultExpiration" value="${redis.expiration}"/>
        <!--是否使用缓存前缀,与cachePrefix相关-->
        <property name="usePrefix" value="true"/>
        <!--配置缓存前缀名称-->
        <property name="cachePrefix">
            <bean class="org.springframework.data.redis.cache.DefaultRedisCachePrefix">
                <constructor-arg index="0" value="-cache-"/>
            </bean>
        </property>
    </bean>
 
    <!--6.配置缓存生成键名的生成规则-->
    <bean id="cacheKeyGenerator" class="com.junlinyi.ssm.redis.CacheKeyGenerator"></bean>
 
    <!--7.启用缓存注解功能-->
    <cache:annotation-driven cache-manager="redisCacheManager" key-generator="cacheKeyGenerator"/>
</beans>

 applicationContext-hiro.xml 構成ファイル  を作成します。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
 
    <!--配置自定义的Realm-->
    <bean id="shiroRealm" class="com.junlinyi.ssm.shiro.MyRealm">
        <property name="userBiz" ref="userBiz" />
        <!--注意:重要的事情说三次~~~~~~此处加密方式要与用户注册时的算法一致 -->
        <!--注意:重要的事情说三次~~~~~~此处加密方式要与用户注册时的算法一致 -->
        <!--注意:重要的事情说三次~~~~~~此处加密方式要与用户注册时的算法一致 -->
        <!--以下三个配置告诉shiro将如何对用户传来的明文密码进行加密-->
        <property name="credentialsMatcher">
            <bean id="credentialsMatcher" class="org.apache.shiro.authc.credential.HashedCredentialsMatcher">
                <!--指定hash算法为MD5-->
                <property name="hashAlgorithmName" value="md5"/>
                <!--指定散列次数为1024次-->
                <property name="hashIterations" value="1024"/>
                <!--true指定Hash散列值使用Hex加密存. false表明hash散列值用用Base64-encoded存储-->
                <property name="storedCredentialsHexEncoded" value="true"/>
            </bean>
        </property>
    </bean>
 
    <!--注册安全管理器-->
    <bean id="securityManager" class="org.apache.shiro.web.mgt.DefaultWebSecurityManager">
        <property name="sessionManager" ref="sessionManager"></property>
        <property name="realm" ref="shiroRealm" />
    </bean>
 
    <!--Shiro核心过滤器-->
    <bean id="shiroFilter" class="org.apache.shiro.spring.web.ShiroFilterFactoryBean">
        <!-- Shiro的核心安全接口,这个属性是必须的 -->
        <property name="securityManager" ref="securityManager" />
        <!-- 身份验证失败,跳转到登录页面 -->
        <property name="loginUrl" value="/login"/>
        <!-- 身份验证成功,跳转到指定页面 -->
        <!--<property name="successUrl" value="/index.jsp"/>-->
        <!-- 权限验证失败,跳转到指定页面 -->
        <property name="unauthorizedUrl" value="/unauthorized.jsp"/>
        <!-- Shiro连接约束配置,即过滤链的定义 -->
        <property name="filterChainDefinitions">
            <value>
                <!--
                注:anon,authcBasic,auchc,user是认证过滤器
                    perms,roles,ssl,rest,port是授权过滤器
                -->
                <!--anon 表示匿名访问,不需要认证以及授权-->
                <!--authc表示需要认证 没有进行身份认证是不能进行访问的-->
                <!--roles[admin]表示角色认证,必须是拥有admin角色的用户才行-->
                /user/login=anon
                /user/updatePwd.jsp=authc
                /admin/*.jsp=roles[4]
                /user/teacher.jsp=perms[2]
                <!-- /css/**               = anon
                 /images/**            = anon
                 /js/**                = anon
                 /                     = anon
                 /user/logout          = logout
                 /user/**              = anon
                 /userInfo/**          = authc
                 /dict/**              = authc
                 /console/**           = roles[admin]
                 /**                   = anon-->
            </value>
        </property>
    </bean>
 
    <!-- Shiro生命周期,保证实现了Shiro内部lifecycle函数的bean执行 -->
    <bean id="lifecycleBeanPostProcessor" class="org.apache.shiro.spring.LifecycleBeanPostProcessor"/>
 
 
    <!-- Session ID 生成器 -->
    <bean id="sessionIdGenerator" class="org.apache.shiro.session.mgt.eis.JavaUuidSessionIdGenerator">
    </bean>
 
    <!--sessionDao自定义会话管理,针对Session会话进行CRUD操作-->
    <bean id="customSessionDao" class="org.apache.shiro.session.mgt.eis.MemorySessionDAO">
        <property name="sessionIdGenerator" ref="sessionIdGenerator"/>
    </bean>
 
    <!--会话监听器-->
    <bean id="shiroSessionListener" class="com.junlinyi.ssm.shiro.MySessionListener"/>
 
    <!--会话cookie模板-->
    <bean id="sessionIdCookie" class="org.apache.shiro.web.servlet.SimpleCookie">
        <!--设置cookie的name-->
        <constructor-arg value="shiro.session"/>
        <!--设置cookie有效时间 永不过期-->
        <property name="maxAge" value="-1"/>
        <!--设置httpOnly 防止xss攻击:cookie劫持-->
        <property name="httpOnly" value="true"/>
    </bean>
 
    <!--SessionManager会话管理器-->
    <bean id="sessionManager" class="org.apache.shiro.web.session.mgt.DefaultWebSessionManager">
        <!--设置session会话过期时间 毫秒 2分钟=120000-->
        <property name="globalSessionTimeout" value="120000"/>
        <!--设置sessionDao-->
        <property name="sessionDAO" ref="customSessionDao"/>
        <!--设置间隔多久检查一次session的有效性 默认1分钟-->
        <property name="sessionValidationInterval" value="60000"/>
        <!--配置会话验证调度器-->
        <!--<property name="sessionValidationScheduler" ref="sessionValidationScheduler"/>-->
        <!--是否开启检测,默认开启-->
        <!--<property name="sessionValidationSchedulerEnabled" value="true"/>-->
        <!--是否删除无效的session,默认开启-->
        <property name="deleteInvalidSessions" value="true"/>
        <!--配置session监听器-->
        <property name="sessionListeners">
            <list>
                <ref bean="shiroSessionListener"/>
            </list>
        </property>
        <!--会话Cookie模板-->
        <property name="sessionIdCookie" ref="sessionIdCookie"/>
        <!--取消URL后面的JSESSIONID-->
        <property name="sessionIdUrlRewritingEnabled" value="true"/>
    </bean>
</beans>

 参照される構成ファイル ( applicationContext.xml など) で上記の構成ファイルを参照します。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
    <!--1. 引入外部多文件方式 -->
    <bean id="propertyConfigurer"
          class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
        <property name="systemPropertiesModeName" value="SYSTEM_PROPERTIES_MODE_OVERRIDE" />
        <property name="ignoreResourceNotFound" value="true" />
        <property name="locations">
            <list>
                <value>classpath:jdbc.properties</value>
                <value>classpath:redis.properties</value>
            </list>
        </property>
    </bean>
 
<!--  框架会越学越多,不能将所有的框架配置,放到同一个配制间,否者不便于管理  -->
    <import resource="applicationContext-mybatis.xml"></import>
    <import resource="spring-redis.xml"></import>
    <import resource="applicationContext-shiro.xml"></import>
</beans>

 上記は基本的な構成手順ですが、特定のプロジェクトの要件やフレームワークのバージョンによって異なる場合があります。実際のアプリケーションでは、システムの可用性と信頼性を向上させるために、必要に応じて Redis のクラスター、センチネル モード、永続化などの機能を構成することもできます。

2. アノテーションの開発と適用シナリオ

2.1. @キャッシュ可能

@Cacheable は Spring Framework のアノテーションで、メソッドの戻り値をキャッシュできることを宣言するために使用されます。これは、メソッドの戻り値がメモリにキャッシュされるため、後続の呼び出しでは、メソッドを再度実行することなく、キャッシュされた結果を直接返すことができることを意味します。

具体的には、@Cacheable アノテーションは次のように機能します。

1. 結果のキャッシュ:@Cacheable アノテーションで変更されたメソッドが呼び出されるとき、Spring はまずメソッドの戻り結果がキャッシュに存在するかどうかを確認します。すでにキャッシュに存在する場合は、メソッド本体のコード ロジックを実行せずに、キャッシュ内の結果が直接返されます。

2. キャッシュ キーの生成:@Cacheableアノテーションでは、キャッシュ内のデータを識別するためのキャッシュ キー (キー) を指定できます。デフォルトでは、キャッシュ キーはメソッドのパラメータで構成されます。 2 つの呼び出しでメソッド パラメータが同じである場合、同じキャッシュ キーが使用され、キャッシュ内の結果が直接返されます。

3. キャッシュ管理:@Cacheable アノテーションは、他のキャッシュ管理ツール (Redis、Ehcache など) と統合できます。構成ファイルで対応するキャッシュ マネージャーを構成することにより、メソッドの戻り結果を、後続の呼び出しのために指定されたキャッシュに保存できます。

4. キャッシュの無効化:@Cacheableアノテーションでは、キャッシュの有効期間を制御するために有効期限 (TTL) を指定することもできます。キャッシュの有効期限が切れると、次にメソッドが呼び出されたときに、メソッド本体のコード ロジックが再実行され、新しい結果がキャッシュに保存されます。
@Cacheable アノテーションを使用すると、データベースやその他のリソースへのアクセス数が効果的に削減され、システムの応答速度と同時処理能力が向上します。。ただし、キャッシュを使用する場合は、データの不整合や有効期限の問題を避けるために、キャッシュの一貫性とリアルタイム パフォーマンスのバランスを取る必要があることに注意してください。

よく使用されるプロパティと使用法

@Cacheable アノテーションには、次の共通の属性と使用関数があります。

 

値:異なるキャッシュ スペースを区別するために使用されるキャッシュの名前を指定します。対応するキャッシュ マネージャーを構成ファイルで構成して、キャッシュを保存するキャッシュ スペースを決定できます。複数のキャッシュ名をカンマで区切って指定できます。

key:キャッシュ内のデータを識別するために使用されるキャッシュ キーを指定します。デフォルトでは、キャッシュ キーはメソッドのパラメータで構成されます。キャッシュ キーは SpEL 式を使用して指定できます。例: @Cacheable(key = "#id") (id はメソッドのパラメータ)。

条件:キャッシュ操作を実行するかどうかを決定する SpEL 式を指定します。キャッシュ操作は、式が true と評価された場合にのみ実行されます。例: @Cacheable(condition = "#result != null")。これは、メソッドの戻り結果が空でない場合にのみキャッシュ操作が実行されることを意味します。

ただし:キャッシュ操作を実行するかどうかを決定する SpEL 式を指定します。キャッシュ操作は、式が false と評価された場合にのみ実行されます。例: @Cacheable(unless = "#result == null") は、メソッドの戻り結果が空の場合にのみキャッシュ操作が実行されないことを意味します。

keyGenerator:キャッシュ キーの生成に使用されるカスタム キャッシュ キー ジェネレーターを指定します。 KeyGenerator インターフェイスを実装して、キャッシュ キー生成ロジックをカスタマイズできます。
@Cacheable アノテーションを使用すると、メソッドの戻り結果をキャッシュできるため、システムの応答速度と同時処理能力が向上します。 。キャッシュ名、キャッシュ キー、条件、キー ジェネレーターなどのプロパティを指定することで、キャッシュをより詳細に制御できます。同時に、データの不整合や有効期限の問題を避けるために、キャッシュの一貫性とリアルタイムのパフォーマンスのバランスに注意を払う必要があります。

基本的な使用例:

@Cacheable(value = "clz" ,key = "'cid'+#cid")

 使用法内のプロパティの説明: 

  • 値: キャッシュの名前を指定するために使用されます。構成ファイルでさまざまなキャッシュ マネージャーを定義できます。ここでの「値」は、使用するキャッシュを指定します。
  • key 属性: SpEL 式「'cid'+#cid」を使用してキャッシュ キーを生成します。このキーは、メソッドの cid パラメータの値から動的に生成されます。メソッドの呼び出し時に異なる cid 値が渡された場合、異なるキャッシュ キーが生成されるため、異なる cid 値が異なるキャッシュ項目に対応します。これにより、キャッシュされたデータをさまざまなコンテキストで保存および取得できるようになります。

2.2. @CachePut 

@CachePut Spring Framework のアノテーションで、メソッドの戻り値をキャッシュに格納するために使用され、通常はキャッシュ内のデータを更新するために使用されます。

@CachePutメソッド関数:

  1. キャッシュ データの更新:@CachePut は、同じキーがすでにキャッシュに存在するかどうかに関係なく、メソッドの戻り値を強制的にキャッシュに保存するために使用されます。これは、特にキャッシュされたデータを手動で更新する必要がある場合に、キャッシュ内のデータを確実に最新にするのに役立ちます。
  2. キャッシュ キーを動的に生成する:@CachePut を使用すると、Spring Expression Language (SpEL) 式を使用してキャッシュ アイテムのキーを動的に生成できます。これにより、メソッド パラメーターやその他の条件に基づいてキャッシュ キーを生成し、異なるキャッシュ アイテムが異なるキーを持つようにすることができます。
  3. 条件付き更新:@CachePut は、条件に基づいてキャッシュ更新操作を実行するかどうかを制御できる、条件属性と Unless 属性をサポートしています。これにより、特定の条件下でのみキャッシュを更新することで、キャッシュをより柔軟に管理できます。
  4. 指定したキャッシュ エントリをクリアする:allEntries 属性を true に設定すると、特定のキャッシュ エントリを更新するだけでなく、指定したキャッシュに関連するすべてのキャッシュ エントリをクリアできます。
  5. 更新タイミングの制御:beforeInvocation 属性を使用すると、キャッシュ更新をメソッドの実行前にトリガーするか、メソッドが正常に実行された後にトリガーするかを制御できます。

注記:

  • @CachePut アノテーションは、キャッシュ内のデータを更新するために使用されます。@Cacheable とは異なり、メソッド本体を実行し、メソッドによって返された値をキャッシュに保存して、キャッシュ内のデータは最新です。
  • キャッシュ内のデータが存在しない場合、@CachePut は新しいキャッシュ エントリを作成します。

 基本的な使用例:

@CachePut(value = "xx",key = "'cid:'+#cid")

使用法内のプロパティの説明: 

  • @CachePutアノテーションはメソッドの戻り値をキャッシュに保存するために使用され、通常はキャッシュ内のデータを更新するために使用されます。
  • @Cacheable とは異なります。同じキーがキャッシュにすでに存在するかどうかはチェックしません。代わりに、メソッドの戻り値をキャッシュに直接保存して、キャッシュ内のデータは最新です。
  • これは、キャッシュ項目を更新して、キャッシュ内のデータがバックエンド データと同期していることを確認する場合に役立ちます。

2.3. @CacheEvict

@CacheEvict Spring Framework のアノテーションで、指定されたキャッシュ項目をキャッシュから削除したり、キャッシュ全体をクリアしたりするために使用されます。

@CacheEvict メソッド関数:

 

  1. 指定したキャッシュ アイテムをクリアする:@CacheEvict を使用して 1 つ以上の特定のキャッシュ内のキャッシュ アイテムをクリアし、キャッシュ内のデータが最新の状態に保たれるようにするか、特定の条件が満たされたときにキャッシュ アイテムをクリアできます。キャッシュをクリアします。
  2. 条件付きクリア:@CacheEvict は、条件に基づいてキャッシュ クリア操作を実行するかどうかを制御できる、条件属性と Unless 属性をサポートしています。これにより、特定の条件が満たされた場合にのみキャッシュをクリアすることで、キャッシュをより柔軟に管理できます。
  3. キャッシュ全体をクリアする:allEntries プロパティを true に設定すると、特定のキャッシュ エントリだけをクリアするのではなく、キャッシュ全体をクリアできます。これは、特定の状況下でキャッシュをグローバルにクリアする必要があるシナリオに役立ちます。
  4. クリアのタイミングの制御:beforeInvocation 属性を使用すると、キャッシュ クリア オペレーションをメソッドの実行前にトリガーするか、メソッドが正常に実行された後にトリガーするかを制御できます。
  5. キャッシュから複数のエントリをクリアする:key 属性を介してキャッシュ キー式を設定し、特定のキー パターンに一致するキャッシュ エントリを削除できます。これにより、特定のパターンに従ってキャッシュされたアイテムを削除できます。

@CacheEvict アノテーションの主な機能は、指定されたキャッシュ アイテムをキャッシュから削除するか、キャッシュ全体をクリアして、キャッシュ内のデータが最新の状態に保たれるようにすることです。または、条件に基づいてキャッシュをクリアします。特定のビジネス ニーズを満たすために、必要に応じてキャッシュをクリアする方法とタイミングを構成できる複数のプロパティが提供されます。

 基本的な使用例:

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

 使用法内のプロパティの説明:

  • value 属性: 「xx」に設定すると、「xx」という名前のキャッシュがクリアされることを示します。通常、構成内で対応するキャッシュ マネージャーを定義して、このキャッシュ名に確実に関連付けられるようにする必要があります。
  • key 属性: SpEL 式「'cid:'+#cid」を使用して、キャッシュ アイテムのキーを生成します。このキーは、「cid:」という接頭辞が付いたメソッドの cid パラメータ値から動的に生成されます。これにより、「cid:」に一致し、その後に cid パラメータ値が続くキャッシュ エントリがクリアされます。
  • allEntries 属性: true に設定すると、キャッシュ全体がクリアされます。 allEntries が true に設定されている場合、key 属性は無視され、指定されたキャッシュ内のすべてのキャッシュされたエントリがクリアされます。この例では、cid パラメータの値に関係なく、「xx」という名前のキャッシュの内容がすべてクリアされます。

2.4. 概要

@Cacheable@CachePut および @CacheEvict は、Spring Framework でキャッシュを管理するために使用されるアノテーションです。これらには、異なる機能と動作があります。

  • @キャッシュ可能:
  1. 関数:@Cacheable は、メソッドの戻り値をキャッシュできることを宣言するために使用されます。つまり、メソッドが呼び出されたとき、Spring は最初にキャッシュをチェックします。対応する結果がキャッシュ内に存在する場合、メソッド本体は実行されずにキャッシュされた値が返されます。
  2. 主な目的:パフォーマンスを向上させ、同じメソッドの繰り返し実行を回避します。読み取り操作に適しており、データの更新には使用されません
  3. 設定:キャッシュ名、キャッシュ キー、条件などを指定できます。
  • @CachePut:
  1. 機能:@CachePut は、メソッドの戻り値を強制的にキャッシュに保存するために使用され、通常はキャッシュ内のデータを更新するために使用されます。
  2. 主な目的:キャッシュを更新し、メソッドから返された値を書き込み操作に適したキャッシュに置きます。
  3. 設定:キャッシュ名、キャッシュ キー、条件などを指定できます。
  • @CacheEvict:
  1. 機能:@CacheEvict は、指定されたキャッシュ アイテムをキャッシュから削除するか、キャッシュ全体をクリアするために使用されます。
  2. 主な目的:キャッシュ アイテムをクリアし、キャッシュ内のデータを最新の状態に保つか、条件に基づいてキャッシュをクリアします。
  3. 設定:キャッシュ名、キャッシュキー、条件、キャッシュ全体をクリアするかどうかなどを指定できます。

違いの概要:

  • @Cacheable メソッドの繰り返し実行を避けるためにメソッドの戻り値をキャッシュするために使用され、読み取り操作に適しています。
  • @CachePut メソッドの戻り値をキャッシュに格納するために使用され、通常はキャッシュ内のデータを更新するために使用され、書き込み操作に適しています。
  • @CacheEvict キャッシュ項目をクリアするために使用されます。指定したキャッシュ項目またはキャッシュ全体をクリアできます。

使用について: 

これらのアノテーションは、特定のビジネス ニーズに応じて組み合わせて使用​​し、柔軟なキャッシュ戦略を実装できます。たとえば、 @Cacheable  を使用して読み取り操作の結果をキャッシュしたり、  を使用してキャッシュ項目を更新したり、  を使用してクリアしたりすることができます。さまざまなキャッシュのニーズを満たすために、キャッシュ内のデータ。  @CachePut @CacheEvict

3. Redis の破壊と侵入雪崩

Redis に関して言えば、キャッシュ システムの高可用性と安定性に関連する「故障」、「侵入」、「雪崩」などの一般的な問題があります。以下では、これら 3 つの質問について詳しく説明します。

  • キャッシュの内訳:

ホットスポットデータが無効な場合に表示されます。特定のキーの有効期限が切れ、多数の同時リクエストが受信されると、これらのリクエストはキャッシュをバイパスしてデータベースに直接アクセスし、データベースの負荷が急激に増加します。キャッシュを無効化すると、次回アクセス時にキャッシュからデータを取得できなくなり、データベースからデータを取得する必要があるためです。

  • キャッシュの侵入:

悪意のあるユーザーがキャッシュまたはデータベースに存在しないキーを要求すると、キャッシュ システムはデータを処理できず、これらの要求はデータベースに直接送信されます。これにより、データベースの負荷が増加したり、サービス拒否攻撃が発生したりする可能性があります。ペネトレーションの問題に対処するには、クエリにキーが存在しない場合に null 値を設定するか、ブルーム フィルターなどの方法を使用して無効なリクエストをフィルター処理します。

  • 雪崩(キャッシュアバランチ):

キャッシュ内の多数のキーが同時に無効になると、多数のリクエストがバックエンド データベースに直接アクセスし、データベースへの負荷が急激に増大し、場合によってはデータベースの崩壊を引き起こします。これは通常、キャッシュ内のデータが同じ有効期限に設定されており、多数のキーが同時に期限切れになり、雪崩現象が発生することが原因です。雪崩の問題を回避するために、ランダムな有効期限やホットスポット データのプリロードの追加などの戦略を採用できます。

これらの問題に対処するには、次のような解決策を講じることができます。

  • 故障の問題に対しては、ミューテックス ロックを使用するか、短い有効期限を設定してホット データを保護し、キャッシュ内に有効なデータが確実に存在するようにすることができます。
  • ペネトレーションの問題については、ブルーム フィルターなどのテクノロジーを使用して無効なリクエストをフィルタリングし、キャッシュ システムが有効なリクエストのみを処理するようにすることができます。
  • アバランシェ問題の場合、分散キャッシュ、マルチレベル キャッシュ、キャッシュの予熱などの戦略を使用して、キャッシュ内のデータの均一な分散を確保し、大量のデータの同時障害を回避できます。

               はい、今日はここまでです! !お役に立てれば! !​ 

おすすめ

転載: blog.csdn.net/m0_74915426/article/details/134353702