一、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设置。
二、cache的缓存结构三、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、从缓存中读取之前缓存存储的数据。
四、Spring缓存存储图例
五、缓存注解
六、缓存参数
七、缓存调用方法
八、源码
Department
package com.citydo.springbootcache.bean;
public class Department {
private Integer id;
private String departmentName;
public Department() {
super();
// TODO Auto-generated constructor stub
}
public Department(Integer id, String departmentName) {
super();
this.id = id;
this.departmentName = departmentName;
}
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getDepartmentName() {
return departmentName;
}
public void setDepartmentName(String departmentName) {
this.departmentName = departmentName;
}
@Override
public String toString() {
return "Department [id=" + id + ", departmentName=" + departmentName + "]";
}
}
Employee
package com.citydo.springbootcache.bean;
import java.io.Serializable;
public class Employee implements Serializable {
private Integer id;
private String lastName;
private String email;
private Integer gender; //性别 1男 0女
private Integer dId;
public Employee() {
super();
}
public Employee(Integer id, String lastName, String email, Integer gender, Integer dId) {
super();
this.id = id;
this.lastName = lastName;
this.email = email;
this.gender = gender;
this.dId = dId;
}
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getLastName() {
return lastName;
}
public void setLastName(String lastName) {
this.lastName = lastName;
}
public String getEmail() {
return email;
}
public void setEmail(String email) {
this.email = email;
}
public Integer getGender() {
return gender;
}
public void setGender(Integer gender) {
this.gender = gender;
}
public Integer getdId() {
return dId;
}
public void setdId(Integer dId) {
this.dId = dId;
}
@Override
public String toString() {
return "Employee [id=" + id + ", lastName=" + lastName + ", email=" + email + ", gender=" + gender + ", dId="
+ dId + "]";
}
}
EmployeeController
package com.citydo.springbootcache.controller;
import com.citydo.springbootcache.Service.EmployeeService;
import com.citydo.springbootcache.bean.Employee;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class EmployeeController {
@Autowired
EmployeeService employeeService;
@GetMapping("/emp/{id}")
public Employee getEmployee(@PathVariable("id") Integer id){
Employee emp = employeeService.getEmp(id);
return emp;
}
}
DepartmentMapper
package com.citydo.springbootcache.mapper;
import com.citydo.springbootcache.bean.Department;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Select;
@Mapper
public interface DepartmentMapper {
@Select("SELECT * FROMdepartment WHERE id=#{id}")
Department getDeptById(Integer id);
}
EmployeeMapper
package com.citydo.springbootcache.mapper;
import com.citydo.springbootcache.bean.Employee;
import org.apache.ibatis.annotations.*;
@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 deletEmpById(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
package com.citydo.springbootcache.Service;
import com.citydo.springbootcache.mapper.DepartmentMapper;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@Service
public class DeptService {
@Autowired
DepartmentMapper departmentMapper;
//@Autowired
}
EmployeeService
package com.citydo.springbootcache.Service;
import com.citydo.springbootcache.bean.Employee;
import com.citydo.springbootcache.mapper.EmployeeMapper;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cache.annotation.Cacheable;
import org.springframework.stereotype.Service;
@Service
public class EmployeeService {
@Autowired
EmployeeMapper employeeMapper;
@Cacheable(cacheNames = "emp")
//@Cacheable(cacheNames = "emp",condition = "#id>0")
// @Cacheable(cacheNames = "emp",condition = "#id>0",unless = "#result==null")
//@Cacheable(cacheNames = "emp",key = "#root.args[0]")//将运行的方法缓存 以后再要相同的数据,直接从缓存获取
//@Cacheable(cacheNames = {"emp","emp"},key = )
public Employee getEmp(Integer id){
System.out.println("查询"+id);
Employee empById = employeeMapper.getEmpById(id);
return empById;
}
}
SpringbootcacheApplication
package com.citydo.springbootcache;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cache.annotation.EnableCaching;
@MapperScan("com.citydo.springbootcache.mapper")
@SpringBootApplication
@EnableCaching
public class SpringbootcacheApplication {
public static void main(String[] args) {
SpringApplication.run(SpringbootcacheApplication.class, args);
}
}
application.properties
spring.datasource.url=jdbc:mysql://localhost:3306/spring_cache
spring.datasource.username=root
spring.datasource.password=123456
#可以省略不写
#spring.datasource.driver-class-name=com.mysql.jdbc.Driver
#开启驼峰命名法匹配规则
mybatis.configuration.map-underscore-to-camel-case=true
#打印sql日志
logging.level.com.citydo.springbootcache.mapper=debug
#debug=true
logging.level.org.springframework.boot.autoconfigure = ERROR