【SpringBoot高级篇】SpringBoot集成cache缓存

【SpringBoot高级篇】SpringBoot集成cache缓存

  • Spring Boot与缓存
    • 一、JSR107
    • 二、Spring缓存抽象
    • 三、几个重要概念&缓存注解
  • 搭建基本环境
    • 导入数据库文件 创建出employee表
    • Pom
    • yml
    • javaBean封装数据
      • Department
      • Employee
    • 整合MyBatis操作数据库
      • DepartmentMapper
      • EmployeeMapper
    • 业务逻辑
      • DeptService
      • EmployeeService
    • controller
      • DeptController
      • EmployeeController
    • 启动类
  • 快速体验缓存
    • 不使用缓存
    • 使用缓存
      • 开启注解缓存 @EnableCaching
      • 标注缓存注解
        • @Cacheable
          • 测试
          • 原理
          • 运行流程
        • @CachePut
        • @CacheEvict
          • 测试
          • 测试
        • @Caching
          • 测试
        • @CacheConfig

学习尚硅谷做的笔记,推荐大家学习

Spring Boot与缓存

一、JSR107

Java Caching定义了5个核心接口,分别是CachingProvider, CacheManager, Cache, Entry Expiry

  • CachingProvider定义了创建、配置、获取、管理和控制多个CacheManager。一个应用可 以在运行期访问多个CachingProvider。
  • CacheManager定义了创建、配置、获取、管理和控制多个唯一命名的Cache,这些Cache 存在于CacheManager的上下文中。一个CacheManager仅被一个CachingProvider所拥有。
  • Cache是一个类似Map的数据结构并临时存储以Key为索引的值。一个Cache仅被一个 CacheManager所拥有。
  • Entry是一个存储在Cache中的key-value对。
  • Expiry 每一个存储在Cache中的条目有一个定义的有效期。一旦超过这个时间,条目为过期 的状态。一旦过期,条目将不可访问、更新和删除。缓存有效期可以通过ExpiryPolicy设置。

【SpringBoot高级篇】SpringBoot集成cache缓存_第1张图片

二、Spring缓存抽象

Spring从3.1开始定义了org.springframework.cache.Cache 和org.springframework.cache.CacheManager接口来统一不同的缓存技术; 并支持使用JCache(JSR-107)注解简化我们开发;

  • Cache接口为缓存的组件规范定义,包含缓存的各种操作集合;
  • Cache接口下Spring提供了各种xxxCache的实现;如RedisCache,EhCacheCache , ConcurrentMapCache等;
  • 每次调用需要缓存功能的方法时,Spring会检查检查指定参数的指定的目标方法是否 已经被调用过;如果有就直接从缓存中获取方法调用后的结果,如果没有就调用方法并缓存结果后返回给用户。下次调用直接从缓存中获取。
  • 使用Spring缓存抽象时我们需要关注以下两点;
  1. 确定方法需要被缓存以及他们的缓存策略
  2. 从缓存中读取之前缓存存储的数据

【SpringBoot高级篇】SpringBoot集成cache缓存_第2张图片

三、几个重要概念&缓存注解

Cache 缓存接口,定义缓存操作。实现有:RedisCache、EhCacheCache、ConcurrentMapCache等
CacheManager 缓存管理器,管理各种缓存(Cache)组件
@Cacheable 主要针对方法配置,能够根据方法的请求参数对其结果进行缓存
@CacheEvict 清空缓存
@CachePut 保证方法被调用,又希望结果被缓存。
@EnableCaching 开启基于注解的缓存
keyGenerator 缓存数据时key生成策略
serialize 缓存数据时value序列化策略

搭建基本环境

导入数据库文件 创建出employee表

/*
Navicat MySQL Data Transfer

Source Server         : 本地
Source Server Version : 50528
Source Host           : 127.0.0.1:3306
Source Database       : springboot_cache

Target Server Type    : MYSQL
Target Server Version : 50528
File Encoding         : 65001
*/

SET FOREIGN_KEY_CHECKS=0;

-- ----------------------------
-- Table structure for department
-- ----------------------------
DROP TABLE IF EXISTS `department`;
CREATE TABLE `department` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `departmentName` varchar(255) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

-- ----------------------------
-- Table structure for employee
-- ----------------------------
DROP TABLE IF EXISTS `employee`;
CREATE TABLE `employee` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `lastName` varchar(255) DEFAULT NULL,
  `email` varchar(255) DEFAULT NULL,
  `gender` int(2) DEFAULT NULL,
  `d_id` int(11) DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

Pom

使用SpringBoot初始化器选择依赖是最好的方式

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-cache</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>2.1.2</version>
        </dependency>

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <scope>runtime</scope>
        </dependency>

yml

spring:
  datasource:
    url: jdbc:mysql://172.16.0.192:3306/springboot_cache
    username: root
    password: 123456
    driver-class-name: com.mysql.cj.jdbc.Driver
#开启驼峰命名 (did不显示)
mybatis:
  configuration:
    map-underscore-to-camel-case: true

javaBean封装数据

Department

@AllArgsConstructor
@NoArgsConstructor
@Data
public class Department implements Serializable {
    private Integer id;
    private String departmentName;

}

Employee

@AllArgsConstructor
@NoArgsConstructor
@Data
public class Employee implements Serializable {
    private Integer id;
    private String lastName;
    private String email;
    private Integer gender; //性别 1男  0女
    private Integer dId;

}

整合MyBatis操作数据库

DepartmentMapper

@Mapper
public interface DepartmentMapper {

    @Select("SELECT * FROM department WHERE id = #{id}")
    Department getDeptById(Integer id);
}

EmployeeMapper

@Mapper
public interface EmployeeMapper {

    @Select("SELECT * FROM employee WHERE id = #{id}")
    public Employee getEmpById(Integer id);

    @Update("UPDATE employee SET lastName=#{lastName},email=#{email},gender=#{gender},d_id=#{dId} WHERE id=#{id}")
    public void updateEmp(Employee employee);

    @Delete("DELETE FROM employee WHERE id=#{id}")
    public void deleteEmpById(Integer id);

    @Insert("INSERT INTO employee(lastName,email,gender,d_id) VALUES(#{lastName},#{email},#{gender},#{dId})")
    public void insertEmployee(Employee employee);

    @Select("SELECT * FROM employee WHERE lastName = #{lastName}")
    Employee getEmpByLastName(String lastName);
}

业务逻辑

DeptService

@Service
public class DeptService {

    @Autowired
    DepartmentMapper departmentMapper;

  
  
    public Department getDeptById(Integer id) {
        System.out.println("查询部门" + id);
        Department department = departmentMapper.getDeptById(id);
        return department;
    }

   
    public Department getDeptByIdManager(Integer id) {
        System.out.println("查询部门" + id);
        Department department = departmentMapper.getDeptById(id);
        return department;
    }


}

EmployeeService

@Service
public class EmployeeService {

    @Autowired
    EmployeeMapper employeeMapper;

  
    public Employee getEmp(Integer id) {
        System.out.println("查询" + id + "号员工");
        Employee emp = employeeMapper.getEmpById(id);
        return emp;
    }


  
    public Employee updateEmp(Employee employee) {
        System.out.println("updateEmp:" + employee);
        employeeMapper.updateEmp(employee);
        return employee;
    }

  
  
    public void deleteEmp(Integer id) {
        System.out.println("deleteEmp:" + id);
        employeeMapper.deleteEmpById(id);
        int i = 10 / 0;
    }

  
    public Employee getEmpByLastName(String lastName) {
        return employeeMapper.getEmpByLastName(lastName);
    }


}

controller

DeptController

@RestController
public class DeptController {

    @Autowired
    DeptService deptService;

    @GetMapping("/dept/{id}")
    public Department getDept(@PathVariable("id") Integer id) {
        return deptService.getDeptById(id);
    }


    @GetMapping("/depts/{id}")
    public Department getDepts(@PathVariable("id") Integer id) {
        return deptService.getDeptByIdManager(id);
    }
}

EmployeeController

@RestController
public class EmployeeController {

    @Autowired
    EmployeeService employeeService;

    @GetMapping("/emp/{id}")
    public Employee getEmployee(@PathVariable("id") Integer id) {
        Employee employee = employeeService.getEmp(id);
        return employee;
    }

    @GetMapping("/emp")
    public Employee update(Employee employee) {
        Employee emp = employeeService.updateEmp(employee);

        return emp;
    }

    @GetMapping("/delemp")
    public String deleteEmp(Integer id) {
        employeeService.deleteEmp(id);
        return "success";
    }

    @GetMapping("/emp/lastname/{lastName}")
    public Employee getEmpByLastName(@PathVariable("lastName") String lastName) {
        return employeeService.getEmpByLastName(lastName);
    }

}

启动类

@MapperScan("cn.zysheep.springboot.mapper")
@SpringBootApplication
@EnableCaching  //开启基于注解的缓存
public class Springboot08CacheApplication {

    public static void main(String[] args) {
        SpringApplication.run(Springboot08CacheApplication.class, args);
    }

}

快速体验缓存

不使用缓存

首页发送请求

BUG:did不显示,在yml里面开启驼峰命名法

后台显示从数据库中查数据

使用缓存

开启注解缓存 @EnableCaching

在启动类中开启注解缓存

@MapperScan("cn.zysheep.springboot.mapper")
@SpringBootApplication
@EnableCaching  //开启基于注解的缓存
public class Springboot08CacheApplication {
    public static void main(String[] args) {
        SpringApplication.run(Springboot08CacheApplication.class, args);
    }
}

标注缓存注解

  • @Cacheable:主要针对方法配置,能够根据方法的请求参数对其结果进行缓存
  • @CacheEvict: 清空缓存
  • @CachePut :保证方法被调用,又希望结果被缓存。
  • @Caching :定义复杂的缓存规则
  • @CacheConfig:抽取缓存的公共配置

@Cacheable

@Cacheable标注的方法执行之前先来检查缓存中有没有这个数据,默认按照参数的值作为key去查询缓存, 如果存在就不再执行该方法,而是直接从缓存中获取结果进行返回,否则才会执行并将返回结果存入指定的缓存中。

public @interface Cacheable {
    @AliasFor("cacheNames")
    String[] value() default {};

    @AliasFor("value")
    String[] cacheNames() default {}; 
    #指定缓存组件的名字;将方法的返回结果放在哪个缓存中,是数组的方式,可以指定多个缓存;

    String key() default "";
    # 缓存数据使用的key;可以用它来指定。默认是使用方法参数的值.(1-方法返回值)
    编写SpEL; #id;参数id的值   #a0  #p0  #root.args[0]

    String keyGenerator() default "";  
    # key的生成器;可以自己指定key的生成器的组件id,key/keyGenerator:二选一使用;

    String cacheManager() default ""; 
    # 指定缓存管理器;或者cacheResolver指定获取解析器

    String cacheResolver() default "";
    # 缓存解析器

    String condition() default ""; 
    # 指定符合条件的情况下才缓存;condition = "#id>0"  condition = "#a0>1":第一个参数的值>1的时候才进行缓存
    String unless() default "";  # 否定缓存;当unless指定的条件为true,方法的返回值就不会被缓存;可以获取到结果进行判断

    boolean sync() default false;  # 是否使用异步模式
}

属性key用SpEL的写法

名字 位置 描述 示例
methodName root object 当前被调用的方法名 #root.methodName
method root object 当前被调用的方法 #root.method.name
target root object 当前被调用的目标对象 #root.target
targetClass root object 当前被调用的目标对象类 #root.targetClass
args root object 当前被调用的方法的参数列表 #root.args[0]
caches root object 当前方法调用使用的缓存列表(如@Cacheable(value={“cache1”, “cache2”})),则有两个cache #root.caches[0].name
argument name evaluation context 方法参数的名字. 可以直接 #参数名 ,也可以使用 #p0或#a0 的形式,0代表参数的索引; #iban 、 #a0 、 #p0
result evaluation context 方法执行后的返回值(仅当方法执行之后的判断有效,如‘unless’,’cache put’的表达式 ’cache evict’的表达式beforeInvocation=false) #result
测试

1)在方法上添加缓存注解

 @Cacheable(value = {"emp"})
 public Employee getEmp(Integer id) {
     System.out.println("查询" + id + "号员工");
     Employee emp = employeeMapper.getEmpById(id);
     return emp;
}

页面访问,首次查询去数据库中看,后台有打印,第二次查同样的数据,会查缓存,后台没有打印

原理

1、自动配置类;CacheAutoConfiguration

2、缓存的配置类

  • org.springframework.boot.autoconfigure.cache.GenericCacheConfiguration
  • org.springframework.boot.autoconfigure.cache.JCacheCacheConfiguration
  • org.springframework.boot.autoconfigure.cache.EhCacheCacheConfiguration
  • org.springframework.boot.autoconfigure.cache.HazelcastCacheConfiguration
  • org.springframework.boot.autoconfigure.cache.InfinispanCacheConfiguration
  • org.springframework.boot.autoconfigure.cache.CouchbaseCacheConfiguration
  • org.springframework.boot.autoconfigure.cache.RedisCacheConfiguration
  • org.springframework.boot.autoconfigure.cache.CaffeineCacheConfiguration
  • org.springframework.boot.autoconfigure.cache.GuavaCacheConfiguration
  • org.springframework.boot.autoconfigure.cache. 【默认】
  • org.springframework.boot.autoconfigure.cache.NoOpCacheConfiguration

3、哪个配置类默认生效:SimpleCacheConfiguration

4、给容器中注册了一个CacheManagerConcurrentMapCacheManager

5、可以获取和创建ConcurrentMapCache类型的缓存组件;他的作用将数据保存在ConcurrentMap中;

运行流程
  1. 方法运行之前,先去查询Cache(缓存组件),按照cacheNames指定的名字获取;CacheManager先获取相应的缓存),第一次获取缓存如果没有Cache组件会自动创建。
  2. 去Cache中查找缓存的内容,使用一个key,默认就是方法的参数;key是按照某种策略生成的;默认是使用keyGenerator生成的,默认使用SimpleKeyGenerator生成key;SimpleKeyGenerator生成key的默认策略;如果没有参数;key=new SimpleKey();如果有一个参数:key=参数的值;如果有多个参数:key=new SimpleKey(params);
  3. 没有查到缓存就调用目标方法;
  4. 将目标方法返回的结果,放进缓存中

@CachePut

@CachePut也可以声明一个方法支持缓存功能。与@Cacheable不同的是使用@CachePut标注的方法在执行前不会去检查缓存中是否存在之前执行过的结果,而是每次都会执行该方法,并将执行结果以键值对的形式存入指定的缓存中。

@CachePut:既调用方法,又更新缓存数据;同步更新缓存 修改了数据库的某个数据,同时更新缓存;

运行时机:

  • 先调用目标方法
  • 将目标方法的结果缓存起来

@CacheEvict

@CacheEvict是用来标注在需要清除缓存元素的方法或类上的。当标记在一个类上时表示其中所有的方法的执行都会触发缓存的清除操作

测试
@CacheEvict(value = "emp", beforeInvocation = true/*key = "#id",*/)
public void deleteEmp(Integer id) {
    System.out.println("deleteEmp:" + id);
    employeeMapper.deleteEmpById(id);
    int i = 10 / 0;
}
  • key:指定要清除的数据
  • allEntries = true:指定清除这个缓存中所有的数据
  • beforeInvocation = false:缓存的清除是否在方法之前执行,默认代表缓存清除操作是在方法执行之后执行;如果出现异常缓存就不会清除
  • beforeInvocation = true:代表清除缓存操作是在方法运行之前执行,无论方法是否出现异常,缓存都清除
测试
  • 查询1号员工;查到的结果会放在缓存中;
  • 以后查询还是之前的结果
  • 更新1号员工;【lastName:zhangsan;gender:0】
  • 查询1号员工?

添加缓存注解

@CachePut(value = "emp")
public Employee updateEmp(Employee employee) {
    System.out.println("updateEmp:" + employee);
    employeeMapper.updateEmp(employee);
    return employee;
}

结果:总是查询一号员工,修改的数据没有查询到

为什么?

因为缓存默认的key为参数,即第一次查询到的emp缓存的key 为id;更新员工将方法的返回值放进缓存了;key是传入的employee对象, 值是返回的employee对象;是两个不同的缓存,所以总是显示为一号员工

如何解决

我们只需要缓存相同的key就可以实现修改缓存数据;即取缓存的key和存缓存的key相同key = "#result.id",key = "#employee.id"

key = "#employee.id":使用传入的参数的员工id;
key = "#result.id":使用返回后的id

注意:@Cacheable的key是不能用#result 为什么是没更新前的?【1号员工没有在缓存中更新】

 @CachePut(value = "emp", key = "#result.id")
    public Employee updateEmp(Employee employee) {
        System.out.println("updateEmp:" + employee);
        employeeMapper.updateEmp(employee);
        return employee;
    }

@Caching

@Caching 定义复杂的缓存规则

测试
@Caching(
            cacheable = {
                    @Cacheable(value="emp",key = "#lastName")
            },
            put = {
                    @CachePut(value="emp",key = "#result.id"),
                    @CachePut(value="emp",key = "#result.email")
            }
    )
    public Employee getEmpByLastName(String lastName) {
   		 return employeeMapper.getEmpByLastName(lastName);
    }

@CacheConfig

每次都要定义cacheNames ,和key,可以在类上定义公共配置,方便管理

@CacheConfig(cacheNames = "emp"/*,cacheManager = "employeeCacheManager"*/) //抽取缓存的公共配置

默认使用的是ConcurrentMapCacheManager==ConcurrentMapCache;将数据保存在 ConcurrentMap

中开发中使用缓存中间件;redis、memcached、ehcache;

你可能感兴趣的:(#,SpringBoot,java,spring,spring,boot)