springboot-cache(redis)

一、搭建基本环境
1、导入数据库文件 创建出department和employee表
2、创建javaBean封装数据
3、整合MyBatis操作数据库
1.配置数据源信息
2.使用注解版的MyBatis;
1)、@MapperScan指定需要扫描的mapper接口所在的包
二、快速体验缓存
步骤:
1、开启基于注解的缓存 @EnableCaching
2、标注缓存注解即可
@Cacheable
@CacheEvict
@CachePut
默认使用的是ConcurrentMapCacheManager==ConcurrentMapCache;将数据保存在 ConcurrentMap<Object, Object>中
开发中使用缓存中间件;redis、memcached、ehcache;


在下面的环境下:
pom:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-cache</artifactId>
</dependency>
@EnableCaching 这个注解是开启缓存
缓存的原理:
1、自动配置类;CacheAutoConfiguration
2、缓存的配置类
在配置脸面加上debug=true,就可以看见那些是不自动配置了
注意: (下面的是所有的缓存自动加载的,在一个环境下只用的到一个,当只引入
spring-boot-starter-cache是加载 SimpleCacheConfiguration,当在引进redis的是,加载的就是
RedisCacheConfiguration(SimpleCacheConfiguration就不会在被加载))由于每个缓存的自动配置类上都有: @ConditionalOnMissingBean(CacheManager.class)
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(这个是默认不加载的)(这个在引入redis的时候会加载,)
org.springframework.boot.autoconfigure.cache.CaffeineCacheConfiguration
org.springframework.boot.autoconfigure.cache.GuavaCacheConfiguration
org.springframework.boot.autoconfigure.cache.SimpleCacheConfiguration【默认】
org.springframework.boot.autoconfigure.cache.NoOpCacheConfiguration
3、哪个配置类默认生效:SimpleCacheConfiguration;
4、给容器中注册了一个CacheManager:ConcurrentMapCacheManager
5、可以获取和创建ConcurrentMapCache类型的缓存组件;他的作用将数据保存在ConcurrentMap中;

SimpleCacheConfiguration的源码:
@Configuration
@ConditionalOnMissingBean(CacheManager.class)
//这个也就限制了只能加载一个xxxCacheConfiguration的原因了,他们是有一定的加载顺序的
@Conditional(CacheCondition.class)
class SimpleCacheConfiguration {
private final CacheProperties cacheProperties;

private final CacheManagerCustomizers customizerInvoker;

SimpleCacheConfiguration(CacheProperties cacheProperties,
CacheManagerCustomizers customizerInvoker) {
this.cacheProperties = cacheProperties;
this.customizerInvoker = customizerInvoker;
}
/**
在默认的情况下是把缓存的数据存在了ConcurrentMapCacheManager这个类里面的
ConcurrentMap<String, Cache> map里面了
*/
@Bean
public ConcurrentMapCacheManager cacheManager() {
ConcurrentMapCacheManager cacheManager = new ConcurrentMapCacheManager();
List<String> cacheNames = this.cacheProperties.getCacheNames();
if (!cacheNames.isEmpty()) {
cacheManager.setCacheNames(cacheNames);
}
return this.customizerInvoker.customize(cacheManager);
}
}
下面是 ConcurrentMapCacheManager
public class ConcurrentMapCacheManager implements CacheManager, BeanClassLoaderAware {

这是从缓存里面获得数据的逻辑:
@Override
public Cache getCache(String name) {
//用key值从map里面取出值
Cache cache = this.cacheMap.get(name);
if (cache == null && this.dynamic) {
//缓存里面没有取出值
synchronized (this.cacheMap) {
cache = this.cacheMap.get(name);
if (cache == null) {
//缓存里面没有的话,就创建一个缓存,在把数据存进去
cache = createConcurrentMapCache(name);
this.cacheMap.put(name, cache);
}
}
}
return cache;
}
下面是往缓存里面存放数据
public void setCacheNames(Collection<String> cacheNames) {
if (cacheNames != null) {
for (String name : cacheNames) {
this.cacheMap.put(name, createConcurrentMapCache(name));
}
this.dynamic = false;
}
else {
this.dynamic = true;
}
}
运行流程:
@Cacheable标注的方法:
1、方法运行之前,先去查询Cache(缓存组件),按照cacheNames指定的名字获取;(CacheManager先获取相应的缓存),第一次获取缓存如果没有Cache组件会自动创建。
CacheAspectSupport(findCachedItem---》execute)---》AbstractCacheResolver---》ConcurrentMapCacheManager
2、去Cache中查找缓存的内容,使用一个key,默认就是方法的参数;key是按照某种策略生成的;默认是使用keyGenerator生成的,默认使用SimpleKeyGenerator生成key;
SimpleKeyGenerator生成key的默认策略;
如果没有参数;key=new SimpleKey();
如果有一个参数:key=参数的值
如果有多个参数:key=new SimpleKey(params);
自定义KeyGenerator:(下面就是自定义的key生成)
@Primary
@Bean
public KeyGenerator myKeyGenerator(){
return new KeyGenerator(){
@Override
public Object generate(Object target, Method method, Object... params) {
String name=target.toString()+method.getName()+Arrays.asList(params).toString();
return name;
}
};
}
3、没有查到缓存就调用目标方法;
4、将目标方法返回的结果,放进缓存中
@Cacheable标注的方法执行之前先来检查缓存中有没有这个数据,默认按照参数的值作为key去查询缓存,
如果没有就运行方法并将结果放入缓存;以后再来调用就可以直接使用缓存中的数据;
核心:
1)、使用CacheManager【ConcurrentMapCacheManager】按照名字得到Cache【ConcurrentMapCache】组件
2)、key使用keyGenerator生成的,默认是SimpleKeyGenerator
@Cacheable的 几个属性
cacheNames/value: 指定缓存组件的名字;将方法的返回结果放在哪个缓存中,是数组的方式,可以指定多个缓存;
key: 缓存数据使用的key;可以用它来指定。默认是使用方法参数的值 1-方法的返回值
编写SpEL; #i d;参数id的值 #a0 #p0 #root.args[0]
getEmp[2]
keyGenerator: key的生成器;可以自己指定key的生成器的组件id
key/keyGenerator:二选一使用;
cacheManager: 指定缓存管理器;或者cacheResolver指定获取解析器
condition: 指定符合条件的情况下才缓存;
condition = "#id>0"
condition = "#a0>1":第一个参数的值》1的时候才进行缓存
unless: 否定缓存;当unless指定的条件为true,方法的返回值就不会被缓存;可以获取到结果进行判断:
unless = "#result == null"
unless = "#a0==2":如果第一个参数的值是2,结果不缓存;
sync: 是否使用异步模式

@CachePut
既调用方法,又更新缓存数据;同步更新缓存;修改了数据库的某个数据,同时更新缓存;
运行时机:
1、先调用目标方法
2、将目标方法的结果缓存起来

测试步骤:
1、查询1号员工;查到的结果会放在缓存中;
key:1 value:lastName:张三
2、以后查询还是之前的结果
3、更新1号员工;【lastName:zhangsan;gender:0】
将方法的返回值也放进缓存了;
key:传入的employee对象 值:返回的employee对象;
4、查询1号员工?
应该是更新后的员工;
key = "#employee.id":使用传入的参数的员工id;
key = "#result.id":使用返回后的id
@Cacheable的key是不能用#result
* 为什么是没更新前的?【1号员工没有在缓存中更新】
key要一样才能更新。
属性和@

@CacheEvict
缓存清除
key: 指定要清除的数据
allEntries = true: 指定清除这个缓存中所有的数据
beforeInvocation = false :缓存的清除是否在方法之前执行默认代表缓存清除操作是在方法执行之后执行;如果出现异常缓存就不会清除
beforeInvocation = true :代表清除缓存操作是在方法运行之前执行,无论方法是否出现异常,缓存都清除

@Caching 定义复杂的缓存规则
例如:
@Caching(
cacheable = {
@Cacheable(/*value="emp",*/key = "#lastName")
},
put = {
@CachePut(/*value="emp",*/key = "#result.id"),
@CachePut(/*value="emp",*/key = "#result.email")
}
)
三、整合redis作为缓存
Redis 是一个开源(BSD许可)的,内存中的数据结构存储系统,它可以用作数据库、缓存和消息中间件。
1、安装redis:使用docker;
2、引入redis的starter
3、配置redis
4、测试缓存
原理:CacheManager===Cache 缓存组件来实际给缓存中存取数据
1)、引入redis的starter,容器中保存的是 RedisCacheManager;
2)、RedisCacheManager 帮我们创建 RedisCache 来作为缓存组件;RedisCache通过操作redis缓存数据的
3)、默认保存数据 k-v 都是Object;利用序列化保存;如何保存为json
1、引入了redis的starter,cacheManager变为 RedisCacheManager;
2、默认创建的 RedisCacheManager 操作redis的时候使用的是 RedisTemplate<Object, Object>
3、RedisTemplate<Object, Object> 是 默认使用jdk的序列化机制
4)、自定义CacheManager;

将方法的运行结果进行缓存;以后再要相同的数据,直接从缓存中获取,不用调用方法;
CacheManager管理多个Cache组件的,对缓存的真正CRUD操作在Cache组件中,每一个缓存组件有自己唯一一个名字;
Redis常见的五大数据类型
String(字符串)、List(列表)、Set(集合)、Hash(散列)、ZSet(有序集合)
stringRedisTemplate.opsForValue()[String(字符串)]
stringRedisTemplate.opsForList()[List(列表)]
stringRedisTemplate.opsForSet()[Set(集合)]
stringRedisTemplate.opsForHash()[Hash(散列)]
stringRedisTemplate.opsForZSet()[ZSet(有序集合)]
下面是上面的具体操作:
在上面的基础上加上pom:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
和上面的差别是:
1、ConcurrentMapCacheManager换成了RedisCacheManager ,redis用
RedisCacheManager来缓存数据。
2、SimpleCacheConfiguration换成了RedisCacheConfiguration,加入redis的,在启动的时候会自动你那个配置RedisCacheConfiguration这个类。
下面是RedisCacheConfiguration的源码:
@Configuration
@AutoConfigureAfter(RedisAutoConfiguration.class)
@ConditionalOnBean(RedisTemplate.class)
@ConditionalOnMissingBean(CacheManager.class)
@Conditional(CacheCondition.class)
class RedisCacheConfiguration {

private final CacheProperties cacheProperties;

private final CacheManagerCustomizers customizerInvoker;

RedisCacheConfiguration(CacheProperties cacheProperties,
CacheManagerCustomizers customizerInvoker) {
this.cacheProperties = cacheProperties;
this.customizerInvoker = customizerInvoker;
}

@Bean //向容器里面注册RedisCacheManager
public RedisCacheManager cacheManager(RedisTemplate<Object, Object> redisTemplate) {
RedisCacheManager cacheManager = new RedisCacheManager(redisTemplate);
cacheManager.setUsePrefix(true);
List<String> cacheNames = this.cacheProperties.getCacheNames();
if (!cacheNames.isEmpty()) {
cacheManager.setCacheNames(cacheNames);
}
return this.customizerInvoker.customize(cacheManager);
}

}
3、CacheManagerCustomizers这个类可以定义一些缓存的规则,
4、我们可以看一下redis的自动配置类RedisAutoConfiguration:

定义一些redis缓存数据的格式:
默认如果保存对象,使用jdk序列化机制,序列化后的数据保存到redis中
怎么样才能将数据以json的方式保存?
(1)自己将对象转为json
(2)redisTemplate默认的序列化规则;改变默认的序列化规则;
下面的就是实现:
@Bean
public RedisTemplate<Object, Object> redisTemplate(
RedisConnectionFactory redisConnectionFactory)
throws UnknownHostException {
//这个的写法可以参考: RedisAutoConfiguration
RedisTemplate<Object, Object> template = new RedisTemplate<Object, Object>();
template.setConnectionFactory(redisConnectionFactory);
Jackson2JsonRedisSerializer< Object > ser = new Jackson2JsonRedisSerializer< Object >( Object .class);
template.setDefaultSerializer(ser);
return template;
}
//CacheManagerCustomizers可以来定制缓存的一些规则
@Primary //将某个缓存管理器作为默认的
@Bean
public RedisCacheManager cacheManager(RedisTemplate<Object, Object> redisTemplate){
RedisCacheManager cacheManager = new RedisCacheManager(redisTemplate);
//key多了一个前缀
//使用前缀,默认会将CacheName作为key的前缀
cacheManager.setUsePrefix(true);
return cacheManager;
}

猜你喜欢

转载自blog.csdn.net/wojiao228925661/article/details/80965297
今日推荐