SpringBoot 缓存(EhCache 2.x 篇)

SpringBoot 缓存

在 Spring Boot中,通过@EnableCaching注解自动化配置合适的缓存管理器(CacheManager),Spring Boot根据下面的顺序去侦测缓存提供者:
* Generic
* JCache (JSR-107)
* EhCache 2.x
* Hazelcast
* Infinispan
* Redis
* Guava
* Simple

关于 Spring Boot 的缓存机制:
高速缓存抽象不提供实际存储,并且依赖于由org.springframework.cache.Cacheorg.springframework.cache.CacheManager接口实现的抽象。 Spring Boot根据实现自动配置合适的CacheManager,只要缓存支持通过@EnableCaching注释启用即可。

Spring Boot 配置 EhCache 2.x

官方文档上对于注解缓存的介绍资料非常之少,往往需要我们自己去了解相应的缓存提供者。我这里主要介绍的是 EhCache .

引入依赖

pom.xml文件中引入以下依赖

      <!--开启 cache 缓存--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-cache</artifactId></dependency><!-- ehcache 缓存 --><dependency><groupId>net.sf.ehcache</groupId><artifactId>ehcache</artifactId></dependency>

引入配置文件 ehcache.xml

resource文件夹下创建文件ehcache.xml,并进行配置:

<?xml version="1.0" encoding="UTF-8"?>
<ehcache xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:noNamespaceSchemaLocation="http://ehcache.org/ehcache.xsd"updateCheck="false"><defaultCacheeternal="false"maxElementsInMemory="1000"overflowToDisk="false"diskPersistent="false"timeToIdleSeconds="0"timeToLiveSeconds="600"memoryStoreEvictionPolicy="LRU" /><!-- 这里的 users 缓存空间是为了下面的 demo 做准备 --><cachename="users"eternal="false"maxElementsInMemory="100"overflowToDisk="false"diskPersistent="false"timeToIdleSeconds="0"timeToLiveSeconds="300"memoryStoreEvictionPolicy="LRU" />
</ehcache>

 ehcache.xml 文件配置详解:

  • diskStore:为缓存路径,ehcache分为内存和磁盘两级,此属性定义磁盘的缓存位置。
  • defaultCache:默认缓存策略,当ehcache找不到定义的缓存时,则使用这个缓存策略。只能定义一个。
  • name:缓存名称。
  • maxElementsInMemory:缓存最大数目
  • maxElementsOnDisk:硬盘最大缓存个数。
  • eternal:对象是否永久有效,一但设置了,timeout将不起作用。
  • overflowToDisk:是否保存到磁盘,当系统当机时
  • timeToIdleSeconds:设置对象在失效前的允许闲置时间(单位:秒)。仅当eternal=false对象不是永久有效时使用,可选属性,默认值是0,也就是可闲置时间无穷大。
  • timeToLiveSeconds:设置对象在失效前允许存活时间(单位:秒)。最大时间介于创建时间和失效时间之间。仅当eternal=false对象不是永久有效时使用,默认是0.,也就是对象存活时间无穷大。
  • diskPersistent:是否缓存虚拟机重启期数据 Whether the disk store persists between restarts of the Virtual Machine. The default value is false.diskSpoolBufferSizeMB:这个参数设置DiskStore(磁盘缓存)的缓存区大小。默认是30MB。每个Cache都应该有自己的一个缓冲区。
  • diskExpiryThreadIntervalSeconds:磁盘失效线程运行时间间隔,默认是120秒。
  • memoryStoreEvictionPolicy:当达到maxElementsInMemory限制时,Ehcache将会根据指定的策略去清理内存。默认策略是LRU(最近最少使用)。你可以设置为FIFO(先进先出)或是LFU(较少使用)。
  • clearOnFlush:内存数量最大时是否清除。
  • memoryStoreEvictionPolicy:可选策略有:LRU(最近最少使用,默认策略)、FIFO(先进先出)、LFU(最少访问次数)。

FIFO,first in first out,先进先出。
LFU, Less Frequently Used,一直以来最少被使用的。如上面所讲,缓存的元素有一个hit属性,hit值最小的将会被清出缓存。
LRU,Least Recently Used,最近最少使用的,缓存的元素有一个时间戳,当缓存容量满了,而又需要腾出地方来缓存新的元素的时候,那么现有缓存元素中时间戳离当前时间最远的元素将被清出缓存。

在主类加上启动注解

在 Spring Boot 主类加上开启缓存的注解@EnableCaching

demo : SpringBoot + EhCache

搭建 Spring Boot 工程

我搭建了一个普通的 SpringBoot 工程,配置了 Druid+MySQL。
并在数据库中创建了 users 表,各字段如下:

字段名 属性
id bigint
uuid varchar
name varchar
age int

用户实体类

User.java

public class User {private long id;private String uuid;private String name;private Integer age;//省略 get、set 及 toString 方法
}

用户数据库操作接口

UserDao.java

@Mapper
public interface UserDao{void delete(String uuid);User update(User user);User findByUuid(String uuid);int save(@Param("user") User user);
}

用户操作Mapper文件

UserMapper.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapperPUBLIC "-//mybatis.org//DTD Mapper 3.0//EN""http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="qg.fangrui.boot.dao.UserDao"><!--目的:为Dao接口方法提供SQL语句--><!--映射实体对象--><resultMap id="UserResultMap" type="qg.fangrui.boot.model.User"><id property="id" column="id" /><result property="uuid" column="uuid" /><result property="name" column="name" /><result property="age" column="age" /></resultMap><insert id="save">INSERT INTO users(name, age, uuid)VALUES (#{user.name}, #{user.age}, #{user.uuid})</insert><select id="findByUuid" resultType="User">SELECT * FROM users WHERE uuid = #{uuid}</select><delete id="delete">DELETE FROM users WHERE uuid = #{uuid}</delete></mapper>

用户操作 service 层

一般情况下,我们在Sercive层进行对缓存的操作。先介绍 Ehcache 在 Spring 中的注解:在支持 Spring Cache 的环境下,
* @Cacheable : Spring在每次执行前都会检查Cache中是否存在相同key的缓存元素,如果存在就不再执行该方法,而是直接从缓存中获取结果进行返回,否则才会执行并将返回结果存入指定的缓存中。
* @CacheEvict : 清除缓存。
* @CachePut : @CachePut也可以声明一个方法支持缓存功能。使用@CachePut标注的方法在执行前不会去检查缓存中是否存在之前执行过的结果,而是每次都会执行该方法,并将执行结果以键值对的形式存入指定的缓存中。
* 这三个方法中都有两个主要的属性:value 指的是 ehcache.xml 中的缓存策略空间;key 指的是缓存的标识,同时可以用 # 来引用参数。

UserService.java

@Service
public class UserService {//这里的单引号不能少,否则会报错,被识别是一个对象private static final String CACHE_KEY = "'user'";private static final String DEMO_CACHE_NAME = "users";@Autowiredprivate UserDao userDao;//删除用户数据@CacheEvict(value = DEMO_CACHE_NAME,key = "'user_'+#uuid")//这是清除缓存public void delete(String uuid){userDao.delete(uuid);}//更新用户数据@CachePut(value = DEMO_CACHE_NAME,key = "'user_'+#user.getUuid()")public User update(User user) throws CacheException{User user1 = userDao.findByUuid(user.getUuid());if (null == user1){throw new  CacheException("Not Find");}user1.setAge(user.getAge());user1.setName(user.getName());return user1;}//查找用户数据@Cacheable(value=DEMO_CACHE_NAME,key="'user_'+#uuid")public User findByUuid(String uuid){//若找不到缓存将打印出提示语句System.err.println("没有走缓存!"+uuid);return userDao.findByUuid(uuid);}//保存用户数据@CacheEvict(value=DEMO_CACHE_NAME,key=CACHE_KEY)public int save(User user){return userDao.save(user);}
}

Controller 类

最后我们创建一个 Controller 来访问我们的缓存。因为我的 SpringBoot 处于 Debug 模式,会将所有的数据库操作打印出来,这样子缓存作用就可一目了然了。
EhcacheController.java

@RestController
public class EhcacheController {private static final Logger logger = LoggerFactory.getLogger(EhcacheController.class);@Autowiredprivate UserService userService;@RequestMapping("/encache")public String EhcacheTest(){logger.debug("进行Encache缓存测试");System.out.println("====生成第一个用户====");User user1 = new User();//生成第一个用户的唯一标识符 UUIDString u1_uuid = UUID.randomUUID().toString();//去掉 UUID 的 - 符号String uuid1 = u1_uuid.substring(0,8)+u1_uuid.substring(9,13)+u1_uuid.substring(14,18)+u1_uuid.substring(19,23)+u1_uuid.substring(24);user1.setName("张三");user1.setAge(18);user1.setUuid(uuid1);if (userService.save(user1) == 0){throw new JdbcException("用户对象插入数据库失败");}//第一次查询System.out.println(userService.findByUuid(user1.getUuid()));//通过缓存查询System.out.println(userService.findByUuid(user1.getUuid()));System.out.println("====修改数据====");User user2 = new User();user2.setName("李四-update");user2.setAge(22);user2.setId(user1.getId());user2.setUuid(user1.getUuid());try {System.out.println(userService.update(user2));} catch (CacheException e){e.printStackTrace();}System.out.println(userService.findByUuid(user2.getUuid()));return "success";}
}

测试

启动 SpringBoot 工程,访问 http://localhost:8080/encache ,并查看控制台打印信息:

由控制台,我们可以清楚到看到,第一次查询用户信息时,工程将用户信息存入缓存中;在第二次查询时,无需访问数据库直接从缓存中获取用户信息。

个人参考项目:

个人参考项目:https://github.com/FunriLy/springboot-study/tree/master/%E6%A1%88%E4%BE%8B9

转载于:https://www.cnblogs.com/MaxElephant/p/8108597.html

spring boot学习(十三)SpringBoot缓存(EhCache 2.x 篇)相关推荐

  1. spring boot学习(2) SpringBoot 项目属性配置

    第一节:项目内置属性 application.properties配置整个项目的,相当于以前的web.xml: 注意到上一节的访问HelloWorld时,项目路径也没有加:直接是http://loca ...

  2. 在Spring Boot中使用数据缓存

    关注公众号[江南一点雨],专注于 Spring Boot+微服务以及前后端分离等全栈技术,定期视频教程分享,关注后回复 Java ,领取松哥为你精心准备的 Java 干货! 春节就要到了,在回家之前要 ...

  3. Spring Boot学习笔记-进阶(3)

    文章目录 Spring Boot学习笔记-进阶(3) 一.Spring Boot与缓存 二.Spring Boot与消息 三.Spring Boot与检索 四.Spring Boot与任务 异步任务 ...

  4. Spring Boot学习笔记-基础(2)

    Spring Boot学习笔记-基础(2) Spring Boot 优点: – 快速创建独立运行的Spring项目以及与主流框架集成 – 使用嵌入式的Servlet容器,应用无需打成WAR包 – st ...

  5. 注解参数获取不到_scm-springboot基于spring boot的统一注解缓存

    scm-springboot 基于spring boot的统一注解缓存,支持mencached.redis.ehcache的缓存无缝切换.支持单个缓存设置过期时间,灵活的key设置规则,采用fastj ...

  6. Spring Boot学习总结(16)——为什么说Java程序员到了必须掌握Spring boot的时候了?

    分享一个大神的人工智能教程.零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到人工智能的队伍中来!点击浏览教程 Spring Boot 2.0 的推出又激起了一阵学习 Spring Boot 热, ...

  7. Spring Boot 集成 Redis 实现缓存机制

    本文章牵涉到的技术点比较多:spring Data JPA.Redis.Spring MVC,Spirng Cache,所以在看这篇文章的时候,需要对以上这些技术点有一定的了解或者也可以先看看这篇文章 ...

  8. 2023最新首发,全网最全 Spring Boot 学习宝典(附思维导图)

    作者:bug菌 博客:CSDN.掘金.infoQ.51CTO等 简介:CSDN/阿里云/华为云/51CTO博客专家,博客之星Top30,掘金年度人气作者Top40,51CTO年度博主Top12,掘金/ ...

  9. Spring Boot 整合Redis 实现缓存

    本文提纲 一.缓存的应用场景 二.更新缓存的策略 三.运行 springboot-mybatis-redis 工程案例 四.springboot-mybatis-redis 工程代码配置详解 运行环境 ...

最新文章

  1. 多年没有管理的技术博客了,即日起开始管理起技术博客
  2. javascript全局观
  3. 【机器学习入门到精通系列】机器学习系统设计(Precision Recall)
  4. 到喜啦携手神策数据,大数据加速产品服务的创新整合发展
  5. 【小白学PyTorch】13.EfficientNet详解及PyTorch实现
  6. 基本数据类型、包装类、String三者之间的相互转换
  7. mysql超大表处理方式是_第29问:MySQL 的复制心跳说它不想跳了
  8. Memcached Java Client API详解
  9. MonkeyDevice Class
  10. solaris常用命令总结
  11. linux下使用ffmpeg下载m3u8视频
  12. 解决WORD输出Html图片Jpg变模糊的问题
  13. 移动应用推广的那些事儿:应用发现篇
  14. 3CDaemon使用教程
  15. 服务器 分辨率问题 显示器不显示不出来,显示器没有最佳分辨率及分辨率调不了的解决方法...
  16. 软件设计师证书的作用有多大?
  17. DELMIA软件:机器人工作区运动包络功能介绍与创建方法
  18. S7-1200 PLC 激活系统时钟存储位后,相应的位没有工作?
  19. matlab里颜色直方图的画法
  20. 入职体检时为什么要检查血常规

热门文章

  1. mysql实战20 | 幻读是什么,幻读有什么问题?
  2. [学习笔记]拉格朗日中值定理
  3. 用gcc/g++编译运行C/C++程序
  4. Java的反射机制(Reflection)
  5. ios UITableview 刷新某一个cell 或 section
  6. 用Topo研发管理系统作需求管理
  7. NTA - ERP远程接入硬件解决方案
  8. 20181207 上课截图
  9. 20.1 shell脚本介绍 20.2 shell脚本结构和执行 20.3 date命令用法 20.4 shell脚本中的变量...
  10. 最长公共子序列(POJ1458)