日常生活中经常会听到缓存这个词,那到底什么是缓存呢?其实缓存就是数据交换的缓冲区(称作Cache),是临时存贮数据(使用频繁的数据)的地方。当用户查询数据,首先在缓存中寻找,如果找到了则直接执行;如果找不到则去数据库中查找。
缓存的本质就是用空间换时间,牺牲数据的实时性,以服务器内存中的数据暂时代替从数据库读取最新的数据,减少数据库IO,减轻服务器压力,减少网络延迟,加快页面打开速度。
优点:
缺点:
内存容量相对硬盘小
缓存中的数据可能与数据库中数据不一致
内存断电就会清空数据,造成数据丢失
一般在远端服务器上,考虑到客户端请求量多,某些数据请求量大,这些热点数据要频繁的从数据库中读取,给数据库造成压力,导致响应客户端较慢。所以,在一些不考虑实时性的数据中,经常将这些数据存在内存中,当请求时候,能够直接读取内存中的数据及时响应。
缓存主要有解决高性能与高并发与减少数据库压力。缓存本质就是将数据存储在内存中,当数据没有发生本质变化的时候,应尽量避免直接连接数据库进行查询,因为并发高时很可能会将数据库压塌,而是应去缓存中读取数据,只有缓存中未查找到时再去数据库中查询,这样就大大降低了数据库的读写次数,增加系统的性能和能提供的并发量。
Redis 是一个高性能的 Key-Value 开源数据库, 是一个非关系型的数据库,是为了解决高并发、高扩展,大数据存储等一系列的问题而产生的数据库解决方案。但它不能替代关系型数据库,只能作为特定环境下的扩充。
Redis 支持的数据结构类型包括:
为了保证读取效率,Redis 把数据对象存储在内存中,并支持周期性的把更新的数据写入磁盘文件中。而且它还提供了交集和并集,以及一些不同方式排序的操作。
redis的特性决定了它的功能,它可以用来做以下这些事情!
缓存穿透: 指查询一个一定不存在的数据,由于缓存是不命中时需要从数据库查询,查不到数据则不写入缓存,这将导致这个不存在的数据每次请求都要到数据库去查询,造成缓存穿透。
几种解决办法:
缓存击穿: 某个 key 非常热点,访问非常频繁,处于集中式高并发访问的情况,当这个 key 在失效的瞬间,大量的请求就击穿了缓存,直接请求数据库,就像是在一道屏障上凿开了一个洞。
几种解决办法:
缓存雪崩: 当缓存服务器重启、或者大量缓存集中在某一个时间段失效,这样在失效的时候,也会给后端系统(比如DB)带来很大压力,造成数据库后端故障,从而引起应用服务器雪崩。
几种解决办法:
使用缓存很大可能导致数据不一致问题,如下:
所以使用缓存时候,应该结合实际情况,考虑缓存的数据是否有一致性需求。
在pom.xml文件中引入Redis依赖,如下
<dependency>
<groupId>org.springframework.bootgroupId>
<artifactId>spring-boot-starter-data-redisartifactId>
dependency>
增加注解@EnableCaching,开启缓存功能,如下:
package com.demo;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cache.annotation.EnableCaching;
@SpringBootApplication
@MapperScan("com.demo")
@EnableCaching
public class DemoApplication {
public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}
}
在application.properties中配置Redis连接信息,如下:
spring:
redis:
# redis库
database: 0
# redis 服务器地址
host: localhost
# redis 端口号
port: 6379
# redis 密码
password:
# 连接超时时间(毫秒)
timeout: 1000
lettuce:
pool:
# 连接池最大链接数(负数表示没有限制)
max-active: 8
# 连接池最大阻塞等待时间(负数表示没有限制)
max-wait: -1
# 连接池最大空闲连接数
max-idle: 8
# 连接池最小空闲连接数
min-idle: 0
新建Redis缓存配置类,如下:
import org.springframework.cache.CacheManager;
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.serializer.*;
import java.time.Duration;
/**
* Redis 配置类
*/
@Configuration
public class RedisConfig {
/**
* 配置缓存管理器
* @param factory Redis 线程安全连接工厂
* @return 缓存管理器
*/
@Bean
public CacheManager cacheManager(RedisConnectionFactory factory) {
// 生成两套默认配置,通过 Config 对象即可对缓存进行自定义配置
RedisCacheConfiguration cacheConfig1 = RedisCacheConfiguration.defaultCacheConfig()
// 设置过期时间 10 分钟
.entryTtl(Duration.ofMinutes(10))
// 设置缓存前缀
.prefixKeysWith("cache:user:")
// 禁止缓存 null 值
.disableCachingNullValues()
// 设置 key 序列化
.serializeKeysWith(keyPair())
// 设置 value 序列化
.serializeValuesWith(valuePair());
RedisCacheConfiguration cacheConfig2 = RedisCacheConfiguration.defaultCacheConfig()
// 设置过期时间 30 秒
.entryTtl(Duration.ofSeconds(30))
.prefixKeysWith("cache:admin:")
.disableCachingNullValues()
.serializeKeysWith(keyPair())
.serializeValuesWith(valuePair());
// 返回 Redis 缓存管理器
return RedisCacheManager.builder(factory)
.withCacheConfiguration("user", cacheConfig1)
.withCacheConfiguration("admin", cacheConfig2)
.build();
}
/**
* 配置键序列化
* @return StringRedisSerializer
*/
private RedisSerializationContext.SerializationPair<String> keyPair() {
return RedisSerializationContext.SerializationPair.fromSerializer(new StringRedisSerializer());
}
/**
* 配置值序列化,使用 GenericJackson2JsonRedisSerializer 替换默认序列化
* @return GenericJackson2JsonRedisSerializer
*/
private RedisSerializationContext.SerializationPair<Object> valuePair() {
return RedisSerializationContext.SerializationPair.fromSerializer(new GenericJackson2JsonRedisSerializer());
}
}
SpringBoot提供了两个bean来操作redis,分别是RedisTemplate
和 StringRedisTemplate
,这两者的主要区别如下:
RedisTemplate
使用的是JdkSerializationRedisSerializer
,存入数据会将数据先序列化成字节数组然后在存入Redis数据库;
StringRedisTemplate
使用的是StringRedisSerializer。
示例如下:
@RestController
public class UserController {
@Autowired
private UserService userServer;
@Autowired
StringRedisTemplate stringRedisTemplate;
/**
* 查询所有课程
*/
@RequestMapping("/allCourses")
public String findAll() {
List<Courses> courses = userServer.findAll();
// 将查询结果写入redis缓存
stringRedisTemplate.opsForValue().set("hot", String.valueOf(courses));
// 读取redis缓存
System.out.println(stringRedisTemplate.opsForValue().get("courses"));
return "ok";
}
}
@Cacheable
可以标记在一个方法上,也可以标记在一个类上。当标记在一个方法上时表示该方法是支持缓存的,当标记在一个类上时则表示该类所有的方法都是支持缓存的。
如果添加了@Cacheable
注解,那么方法被调用后,值会被存入redis,下次再调用的时候会直接从redis中取值返回。
@Service
@CacheConfig(cacheNames = "user")
public class UserService {
@Autowired
private UserMapper userMapper;
// 获取全部用户
@Cacheable(key = "'allUsers'", unless = "#result==null")
public List<Courses> findAll() {
return userMapper.allUsers();
}
}
1、创建实体类(model层)
public class User {
private String username;
private String password;
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
}
2、创建接口(mapper层)
@Component
public interface UserMapper {
// 查询所有用户
@Select("select * from user_tbl")
List<Courses> allUsers();
// 更新用户信息
@Update("update user_tbl set password=#{password} where username=#{username};")
boolean updateUser(String username, String password);
// 删除用户
@Select("delete from user_tbl where username = #{username};")
Integer delUser(String username);
}
3、创建服务类(service层)
@Service
@CacheConfig(cacheNames = "user")
public class UserService {
@Autowired
private UserMapper userMapper;
// 查询全部用户
@Cacheable(key = "'allUsers'", unless = "#result==null")
public List<Courses> allUsers() {
return userMapper.allUsers();
}
// 更新用户信息
@CachePut(key = "#user.username")
public void updateUser(String username, String password) {
userMapper.updateUser(username, password);
}
// 删除用户
@CacheEvict(key = "#username")
public void delUser(String username) {
userMapper.delUser(username);
}
}
4、创建控制器(controller层)
@RestController
public class UserController {
@Autowired
private UserService userServer;
@Autowired
StringRedisTemplate stringRedisTemplate;
/**
* 查询所有用户
*/
@RequestMapping("/allUsers")
public String allUsers() {
userServer.allUsers();
return "ok";
}
/**
* 更新用户信息
*/
@RequestMapping("/updateUser")
public String updateUser() {
String username = "tom";
String password = "abc123";
userServer.updateUser(username,password);
return "ok";
}
/**
* 删除用户
*/
@RequestMapping("/delUser")
public String delUser() {
String username = "tom";
userServer.delUser(username);
return "ok";
}
}