目录
SpringBoot整合Redis
引入依赖
<!-- redis -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<!-- spring2.X集成redis所需common-pool2 -->
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-pool2</artifactId>
<version>2.6.0</version>
</dependency>
添加redis配置类
import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.cache.CacheManager;
import org.springframework.cache.annotation.CachingConfigurerSupport;
import org.springframework.cache.annotation.EnableCaching;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.cache.RedisCacheConfiguration;
import org.springframework.data.redis.cache.RedisCacheManager;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.RedisSerializationContext;
import org.springframework.data.redis.serializer.RedisSerializer;
import org.springframework.data.redis.serializer.StringRedisSerializer;
import java.time.Duration;
@EnableCaching // 开启缓存
@Configuration // 配置类
public class RedisConfig extends CachingConfigurerSupport {
@Bean
public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) {
RedisTemplate<String, Object> template = new RedisTemplate<>();
RedisSerializer<String> redisSerializer = new StringRedisSerializer();
Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
ObjectMapper om = new ObjectMapper();
om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
jackson2JsonRedisSerializer.setObjectMapper(om);
template.setConnectionFactory(factory);
//key序列化方式
template.setKeySerializer(redisSerializer);
//value序列化
template.setValueSerializer(jackson2JsonRedisSerializer);
//value hashmap序列化
template.setHashValueSerializer(jackson2JsonRedisSerializer);
return template;
}
@Bean
public CacheManager cacheManager(RedisConnectionFactory factory) {
RedisSerializer<String> redisSerializer = new StringRedisSerializer();
Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
//解决查询缓存转换异常的问题
ObjectMapper om = new ObjectMapper();
om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
jackson2JsonRedisSerializer.setObjectMapper(om);
// 配置序列化(解决乱码的问题),过期时间600秒
RedisCacheConfiguration config = RedisCacheConfiguration.defaultCacheConfig()
.entryTtl(Duration.ofSeconds(600))
.serializeKeysWith(RedisSerializationContext.SerializationPair.fromSerializer(redisSerializer))
.serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(jackson2JsonRedisSerializer))
.disableCachingNullValues();
RedisCacheManager cacheManager = RedisCacheManager.builder(factory)
.cacheDefaults(config)
.build();
return cacheManager;
}
}
在接口中添加redis缓存
-
缓存@Cacheable:根据方法对其返回结果进行缓存,下次请求时,如果缓存存在,则直接读取缓存数据返回,如果不存在,则执行方法,并把返回的结果存入缓存中,一般用在查询方法上
属性/方法 解释 value 缓存名,必填,它指定了缓存存放在哪块命名空间 cacheNames 与value差不多 二选一即可 key 可选属性,可以使用SpEL,标签自定义缓存的key -
缓存@CachePut:使用该注解标志的方法,每次都会执行,并将结果存入指定的缓存中。其他方法可以直接从响应的缓存中读取缓存数据,而不需要再去查询数据库。一般用在新增方法上
属性/方法 解释 value 缓存名,必填。它指定了缓存存放在哪块命名空间 cacheNames 与value差不多 二选一即可 key 可选属性,可以使用SpEL,标签自定义缓存的key -
缓存@CacheEvict:使用该注解标志的方法,会清空指定的缓存,一般用在更新或者删除方法上
属性/方法 解释 value 缓存名,必填。它指定了缓存存放在哪块命名空间 cacheNames 与value差不多 二选一即可 key 可选属性,可以使用SpEL,标签自定义缓存的key allEntries 是否清空所有缓存,默认为false。如果指定为true,则方法调用后将立即清空所有缓存 beforeInvocation 是否在执行前就清空,默认为false,如果指定为true,则在方法执行前就会清空缓存
在配置文件中配置redis地址
spring.redis.host=118.178.59.51
spring.redis.port=6379
spring.redis.database=0
spring.redis.timeout=1800000
spring.redis.lettuce.pool.max-active=20
spring.redis.lettuce.pool.max-wait=-1
# 最大阻塞等待时间(复数表示没有限制)
spring.redis.lettuce.pool.max-idle=5
spring.redis.lettuce.pool.min-idle=0
Springboot整合redis实现缓存
引入依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-cache</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
配置RedisConfig
@EnableCaching
@Configuration
public class RedisConfig {
@Bean
RedisCacheConfiguration redisCacheConfiguration() {
// 读取redis的默认配置模板
RedisCacheConfiguration config = RedisCacheConfiguration.defaultCacheConfig();
// 设置过期时间
config = config.entryTtl(Duration.ofDays(1));
// 设置KEY的前缀
config.prefixKeysWith("CACHE_");
// 开启使用KEY前缀
config.usePrefix();
// 设置允许缓存空值,防止缓存穿透
config.getAllowCacheNullValues();
// 配置key的序列化方式
config.serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(new StringRedisSerializer()));
// 配置value的序列化方式
config = config.serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(new GenericJackson2JsonRedisSerializer()));
return config;
}
}
RedisTemplate常用方法
redisTemplate.opsForValue();//操作字符串
redisTemplate.opsForHash();//操作hash
redisTemplate.opsForList();//操作list
redisTemplate.opsForSet();//操作set
redisTemplate.opsForZSet();//操作有序set
Redis常用缓存注解
注解名 作用
@Cacheable 主要针对方法配置,能够根据方法的请求参数对其结果进行缓存
@CacheEvict 清空缓存
@CachePut 保证方法被调用,又希望结果被缓存
@EnableCaching 开启基于注解的缓存
解决缓存穿透、缓存雪崩、缓存击穿
- 空结果缓存:解决缓存穿透问题
- 设置过期时间加随机值:解决缓存雪崩
- 加锁:解决缓存击穿
Redisson整合 可重入锁 读写锁
官网地址 github redisson wiki
导入依赖
<dependency>
<groupId>org.redisson</groupId>
<artifactId>redisson-spring-boot-starter</artifactId>
<!--适用于2.2.x版本-->
<version>3.13.1</version>
</dependency>
配置RedissonConfig
@Configuration
public class RedissonConfig {
@Bean
public RedissonClient redisson() {
//1. 创建配置
Config config = new Config();
config.useSingleServer().setAddress("redis://192.168.56.10:6379");
// 2. 根据config创建出RedissonClient实例
RedissonClient redisson = Redisson.create(config);
return redisson;
}
}
可重入锁
RLock rLock = redissonClient.getLock("lock");
rLock.lock();
rLock.unlock();
读写锁
// 读
@GetMapping("read")
public R read() {
RLock lock = redissonClient.getReadWriteLock("rw-lock").readLock();
try {
lock.lock();
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
lock.unlock();
}
return R.ok().data("data", "read");
}
// 写
@GetMapping("write")
public R write() {
RLock lock = redissonClient.getReadWriteLock("rw-lock").writeLock();
try {
lock.lock();
Thread.sleep(8000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
lock.unlock();
}
return R.ok().data("data", "write");
}