‘
Spring Boot 缓存技术
一、Spring Boot整合Ehcache
1 修改 pom 文件
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.5.10.RELEASE</version>
</parent>
<groupId>com.xk</groupId>
<artifactId>spring-boot-demo</artifactId>
<version>0.0.1-SNAPSHOT</version>
<properties>
<java.version>1.7</java.version>
<thymeleaf.version>3.0.2.RELEASE</thymeleaf.version>
<thymeleaf-layout-dialect.version>2.0.4</thymeleaf-layout-dialect.version>
</properties>
<dependencies>
<!-- springBoot 的启动器 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!-- thymeleaf 的启动器 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>
<!-- springBoot 的启动器 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<!-- 测试工具的启动器 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
</dependency>
<!-- mysql -->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
</dependency>
<!-- druid 连接池 -->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid</artifactId>
<version>1.0.9</version>
</dependency>
<!-- Spring Boot 缓存支持启动器 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-cache</artifactId>
</dependency>
<!-- Ehcache -->
<dependency>
<groupId>net.sf.ehcache</groupId>
<artifactId>ehcache</artifactId>
</dependency>
</dependencies>
</project>
2、创建 Ehcache 的配置文件
文件名:ehcache.xml 位置:src/main/resources/ehcache.xml
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:noNamespaceSchemaLocation="../config/ehcache.xsd">
<diskStore path="java.io.tmpdir" />
<!--defaultCache:echcache的默认缓存策略 -->
<defaultCache maxElementsInMemory="10000" eternal="false"
timeToIdleSeconds="120" timeToLiveSeconds="120"
maxElementsOnDisk="10000000" diskExpiryThreadIntervalSeconds="120"
memoryStoreEvictionPolicy="LRU">
<persistence strategy="localTempSwap" />
</defaultCache>
<!-- 自定义缓存策略 -->
<cache name="users" maxElementsInMemory="10000" eternal="false"
timeToIdleSeconds="120" timeToLiveSeconds="120"
maxElementsOnDisk="10000000" diskExpiryThreadIntervalSeconds="120"
memoryStoreEvictionPolicy="LRU">
<persistence strategy="localTempSwap" />
</cache>
</ehcache>
3、修改 application.properties 文件
spring.datasource.driverClassName=com.mysql.jdbc.Driver spring.datasource.url=jdbc:mysql://localhost:3306/ssm spring.datasource.username=root spring.datasource.password=root
spring.datasource.type=com.alibaba.druid.pool.DruidDataSource
spring.jpa.hibernate.ddl-auto=update spring.jpa.show-sql=true
spring.cache.ehcache.cofnig=ehcache.xml
4 修改启动类
@EnableCaching
@SpringBootApplication
public class App {
public static void main(String[] args) {
SpringApplication.run(App.class, args);
}
}
5、创建业务层
/**
* UsersService 接口实现类 *
*
*/
@Service
public class UsersServiceImpl implements UsersService {
@Autowired
private UsersRepository usersRepository;
@Override
public List<Users> findUserAll() {
return this.usersRepository.findAll();
}
@Override
//@Cacheable:对当前查询的对象做缓存处理
@Cacheable(value = "users")
public Users findUserById(Integer id) {
return this.usersRepository.findOne(id);
}
@Override
public Page<Users> findUserByPage(Pageable pageable) {
return this.usersRepository.findAll(pageable);
}
@Override
public void saveUsers(Users users) {
this.usersRepository.save(users);
}
}
6 修改实体类 Users
@Entity
@Table(name = "t_users")
public class Users implements Serializable {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "id")
private Integer id;
@Column(name = "name")
private String name;
@Column(name = "age")
private Integer age;
@Column(name = "address")
private String address;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
public String getAddress() {
return address;
}
public void setAddress(String address) {
this.address = address;
}
@Override
public String toString() {
return "Users [id=" + id + ", name=" + name + ", age=" + age + ", address=" + address + "]";
}
}
7 测试
/**
* UsersService 测试 *
*
*/
@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = App.class)
public class UsersServiceTest {
@Autowiredprivate
UsersService usersService;
@Test
public void testFindUserById() {
// 第一次查询
System.out.println(this.usersService.findUserById(1));
//第二次查询
System.out.println(this.usersService.findUserById(1));
}
}
结果就是应该执行两次sql的查询语句,但是第二查询并没有执行,原因是因为从缓存里面取出类第一次查询的结果。
二、 @Cacheable 与@CacheEvict
1 @Cacheable
@Cacheable 作用:把方法的返回值添加到 Ehcache 中做缓存
Value 属性:指定一个 Ehcache 配置文件中的缓存策略,如果么有给定 value,name 则 表示使用默认的缓存策略。
Key 属性:给存储的值起个名称。在查询时如果有名称相同的,那么则知己从缓存中将 数据返回
1、业务层
@Override
@Cacheable(value = "users", key = "#pageable.pageSize")
public Page<Users> findUserByPage(Pageable pageable) {
return this.usersRepository.findAll(pageable);
}
2、测试代码
@Test
public void testFindUserByPage() {
Pageable pageable = new PageRequest(0, 2); // 第一次查询
System.out.println(this.usersService.findUserByPage(pageable).getTotalElements());
// 第二次查询
System.out.println(this.usersService.findUserByPage(pageable).getTotalElements());
// 第三次查询
pageable = new PageRequest(1, 2);
System.out.println(this.usersService.findUserByPage(pageable).getTotalElements());
}
这里结果就是第一次查询,第二次没有sql语句查询,直接从缓存里面取,第三次执行类sql查询。
2 @CacheEvict
@CacheEvict 作用:清除缓存
@CacheEvict(value="users",allEntries=true)
public void saveUsers(Users users) {
this.usersRepository.save(users);
}
使用场景,比喻说,我们第一次查询数据库前两条数据添加类缓存,在我们将数据添加到数据库后,数据库对应前两条数据和缓存里面到数据就不一样类,这时我们要清除缓存,查询时重新添加缓存。
三、 Spring Boot 整合 Spring Data Redis
1、Linux安装 Redis
可以下载一个虚拟机进行测试 安装步骤省略,可以在网上找教程,写几个命令,面试用
1、安装 gcc
Yum install gcc-c++
2、解压 redis 压缩包
tar -zxvf redis-3.0.0.tar.gz
3、进入解压后的目录进行编译
cd redis-3.0.0 make
4、将 Redis 安装到指定目录
make PREFIX=/usr/local/redis install
5、启动 Redis
./redis-server
2、Spring Boot 整合 Spring Data Redis
Spring Data Redis 是属于 Spring Data 下的一个模块。作用就是简化对于 redis 的操做
1、修改 pom 文件添加 Spring Data Redis 的坐标
<!-- Spring Data Redis 的启动器 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<!-- Test 的启动器 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
</dependency>
2、编写 Spring Data Redis 的配置类
/**
* 完成对 Redis 的整合的一些配置 *
*/
@Configuration
public class RedisConfig {
/**
* 1.创建 JedisPoolConfig 对象。在该对象中完成一些链接池配置 *
*/
@Bean
public JedisPoolConfig jedisPoolConfig() {
JedisPoolConfig config = new JedisPoolConfig();
// 最大空闲数
config.setMaxIdle(10);
// 最小空闲数
config.setMinIdle(5);
// 最大链接数
config.setMaxTotal(20);
return config;
}
/**
* 2.创建 JedisConnectionFactory:配置 redis 链接信息
*/
@Bean
public JedisConnectionFactory jedisConnectionFactory(JedisPoolConfig config) {
JedisConnectionFactory factory = new JedisConnectionFactory();
// 关联链接池的配置对象
factory.setPoolConfig(config);
// 配置链接 Redis 的信息
// 主机地址
factory.setHostName("192.168.0.106");
// 端口
factory.setPort(6379);
return factory;
}
/**
* 3.创建 RedisTemplate:用于执行 Redis 操作的方法
*/
@Bean
public RedisTemplate<String, Object> redisTemplate(JedisConnectionFactory factory) {
RedisTemplate<String, Object> template = new RedisTemplate<>(); // 关联
template.setConnectionFactory(factory);
// 为 key 设置序列化器
template.setKeySerializer(new StringRedisSerializer());
// 为 value 设置序列化器
template.setValueSerializer(new StringRedisSerializer());
return template;
}
}
3、编写测试代码,测试整合环境
/**
* Spring Data Redis 测试 *
*
*/
@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = App.class)
public class RedisTest {
@Autowired
private RedisTemplate<String, Object> redisTemplate;
/**
* 添加一个字符串
*/
@Test
public void testSet() {
this.redisTemplate.opsForValue().set("key", "北京尚学堂");
}
/**
* 获取一个字符串
*/
@Test
public void testGet() {
String value = (String) this.redisTemplate.opsForValue().get("key");
System.out.println(value);
}
}
3、提取 redis 的配置信息
1、在 src/main/resource/ 目录下新建一个配置文件:
application.properties
spring.redis.pool.max-idle=10
spring.redis.pool.min-idle=5
spring.redis.pool.max-total=20
spring.redis.hostName=192.168.0.106
spring.redis.port=6379
2、修改配置类
/**
* 完成对 Redis 的整合的一些配置 *
*/
@Configuration
public class RedisConfig {
/**
* 1.创建 JedisPoolConfig 对象。在该对象中完成一些链接池配置 *
*/
@Bean
@ConfigurationProperties(prefix = "spring.redis.pool")
public JedisPoolConfig jedisPoolConfig() {
JedisPoolConfig config = new JedisPoolConfig();
// 最大空闲数
// config.setMaxIdle(10);
// 最小空闲数
// config.setMinIdle(5);
// 最大链接数
// config.setMaxTotal(20);
System.out.println("默认值:" + config.getMaxIdle());
System.out.println("默认值:" + config.getMinIdle());
System.out.println("默认值:" + config.getMaxTotal());
return config;
}
/**
* 2.创建 JedisConnectionFactory:配置 redis 链接信息
*/
@Bean
@ConfigurationProperties(prefix = "spring.redis")
public JedisConnectionFactory jedisConnectionFactory(JedisPoolConfig config) {
System.out.println("配置完毕:" + config.getMaxIdle());
System.out.println("配置完毕:" + config.getMinIdle());
System.out.println("配置完毕:" + config.getMaxTotal());
JedisConnectionFactory factory = new JedisConnectionFactory();
// 关联链接池的配置对象
factory.setPoolConfig(config);
// 配置链接 Redis 的信息
// 主机地址
//factory.setHostName("192.168.0.106");
// 端口
//factory.setPort(6379);
return factory;
}
/**
* 3.创建 RedisTemplate:用于执行 Redis 操作的方法
*/
@Bean
public RedisTemplate<String, Object> redisTemplate(JedisConnectionFactory factory) {
RedisTemplate<String, Object> template = new RedisTemplate<>(); // 关联
template.setConnectionFactory(factory);
// 为 key 设置序列化器
template.setKeySerializer(new StringRedisSerializer());
// 为 value 设置序列化器
template.setValueSerializer(new StringRedisSerializer());
return template;
}
}
4 Spring Data Redis 操作实体对象
public class Users implements Serializable {
private Integer id;
private String name;
private Integer age;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
@Override
public String toString() {
// TODO Auto-generated method stub
return "Users [id=" + id + ", name=" + name + ", age=" + age +"]";
}
}
实现序列化接口Serializable
测试
@Test
public void testSetUesrs() {
Users users = new Users();
users.setAge(20);
users.setName("json");
users.setId(1); // 重新设置序列化器
this.redisTemplate.setValueSerializer(new JdkSerializationRedisSerializer());
this.redisTemplate.opsForValue().set("users", users);
}
/**
* 取 Users 对象
*/
@Test
public void testGetUsers() { // 重新设置序列化器
this.redisTemplate.setValueSerializer(new JdkSerializationRedisSerializer());
Users users = (Users) this.redisTemplate.opsForValue().get("users");
System.out.println(users);
}
Spring Data Redis 以 JSON 格式存储实体对象
/**
* 基于 JSON 格式存 Users 对象
*/
@Test
public void testSetUsersUseJSON() {
Users users = new Users();
users.setAge(20);
users.setName("json");
users.setId(1);
this.redisTemplate.setValueSerializer(new Jackson2JsonRedisSerializer<>(Users.class));
this.redisTemplate.opsForValue().set("users_json", users);
}
/**
* 基于 JSON 格式取 Users 对象
*/
@Test
public void testGetUseJSON() {
this.redisTemplate.setValueSerializer(new Jackson2JsonRedisSerializer<>(Users.class));
Users users = (Users) this.redisTemplate.opsForValue().get("users_json");
System.out.println(users);
}
‘
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/4517.html