Springboot25 之搭建ehcache

那么我们先说说这一篇文章我们都会学到的技术点:
Spring Data JPA,Spring Boot 使用Mysql,Spring MVC,EHCache,Spring Cache等
如下几个步骤:
(1)新建Maven Java Project
(2)在pom.xml中加入依赖包
(3)编写Spring Boot启动类;
(4)配置application.properties;
(5)编写缓存配置类以及ehcache.xml配置文件;
(6)编写DemoInfo实体类进行测试;
(7)编写持久类DemoInfoRepository;
(8)编写处理类DemoInfoService;
(9)编写DemoInfoController测试类;
(10)运行测试

1、新建Maven Java Project

2、在pom.xml中加入依赖包

<properties>
  <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
<parent>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-parent</artifactId>
  <version>1.4.1.RELEASE</version>
</parent>
<dependencies>
  <dependency>
    <groupId>junit</groupId>
    <artifactId>junit</artifactId>
    <version>3.8.1</version>
    <scope>test</scope>
  </dependency>
  <!-- spring boot web支持:mvc,aop... -->
  <dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
  </dependency>
  <!--
      包含支持UI模版(Velocity,FreeMarker,JasperReports),
      邮件服务,
      脚本服务(JRuby),
      缓存Cache(EHCache),
      任务计划Scheduling(uartz)。
   -->
  <dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-context-support</artifactId>
  </dependency>
  <!-- 集成ehcache需要的依赖-->
  <dependency>
    <groupId>net.sf.ehcache</groupId>
    <artifactId>ehcache</artifactId>
  </dependency>
  <!-- JPA操作数据库. -->
  <dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-jpa</artifactId>
  </dependency>
  <!-- mysql 数据库驱动. -->
  <dependency>
    <groupId>mysql</groupId>
    <artifactId>mysql-connector-java</artifactId>
  </dependency>
  <!-- Spring boot单元测试. -->
  <dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-test</artifactId>
    <scope>test</scope>
  </dependency>
</dependencies>

3、编写Spring Boot启动类(这里不做说明);
4、配置application.properties(主要JPA,和数据库,具体查看之前的文档);
5、编写缓存配置类以及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">
   <!--
    diskStore:为缓存路径,ehcache分为内存和磁盘两级,此属性定义磁盘的缓存位置。参数解释如下:
       user.home – 用户主目录
       user.dir  – 用户当前工作目录
       java.io.tmpdir – 默认临时文件路径
     -->
   <diskStore path="java.io.tmpdir/Tmp_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,最近最少使用的,缓存的元素有一个时间戳,当缓存容量满了,而又需要腾出地方来缓存新的元素的时候,那么现有缓存元素中时间戳离当前时间最远的元素将被清出缓存。
   -->
   <defaultCache
         eternal="false"
         maxElementsInMemory="1000"
         overflowToDisk="false"
         diskPersistent="false"
         timeToIdleSeconds="0"
         timeToLiveSeconds="600"
         memoryStoreEvictionPolicy="LRU" />
   <cache
         name="demo"
         eternal="false"
         maxElementsInMemory="100"
         overflowToDisk="false"
         diskPersistent="false"
         timeToIdleSeconds="0"
         timeToLiveSeconds="300"
         memoryStoreEvictionPolicy="LRU" />
</ehcache>
@Configuration
@EnableCaching//标注启动缓存
public class CacheConfiguration {
    /**
     *  ehcache 主要的管理器
     * @param bean
     * @return
     */
    @Bean
    public EhCacheCacheManager ehCacheCacheManager(EhCacheManagerFactoryBean bean){
        System.out.println("CacheConfiguration.ehCacheCacheManager()");
        return new EhCacheCacheManager(bean.getObject());
    }
    /*
       * 据shared与否的设置,
       * Spring分别通过CacheManager.create()
       * 或new CacheManager()方式来创建一个ehcache基地.
       *
       * 也说是说通过这个来设置cache的基地是这里的Spring独用,还是跟别的(如hibernate的Ehcache共享)
       *
       */
    @Bean
    public EhCacheManagerFactoryBean ehCacheManagerFactoryBean(){
        System.out.println("CacheConfiguration.ehCacheManagerFactoryBean()");
        EhCacheManagerFactoryBean cacheManagerFactoryBean = new EhCacheManagerFactoryBean ();
        cacheManagerFactoryBean.setConfigLocation (new ClassPathResource("ehcache.xml"));
        cacheManagerFactoryBean.setShared(true);
        return cacheManagerFactoryBean;
    }
}写DemoInfo实体类进行测试;

@Entity
public class DemoInfo {
    @Id
    @GeneratedValue
    private long id;//主键.
    private String name;//名称;
    private String pwd;//密码;
    private int state;
    {get and set方法}
    @Override
    public String toString() {
        return "DemoInfo [id=" + id + ", name=" + name + ", pwd=" + pwd + ", state=" + state + "]";
    }
}

7、编写持久类DemoInfoRepository;
public interface DemoInfoRepository extends CrudRepository<DemoInfo, Long> {
}
8、编写处理类DemoInfoService;
@Component
public class DemoInfoServiceImpl implements DemoInfoService {
   //这里的单引号不能少,否则会报错,被识别是一个对象;
    public static final String CACHE_KEY = "'demoInfo'";

    @Resource
    private DemoInfoRepository demoInfoRepository;

    /**
     * value属性表示使用哪个缓存策略,缓存策略在ehcache.xml
     */
    public static final String DEMO_CACHE_NAME = "demo";

    /**
     * 保存数据.
     * @param demoInfo
     */
    @CacheEvict(value=DEMO_CACHE_NAME,key=CACHE_KEY)
    @Override
    public DemoInfo save(DemoInfo demoInfo){
        return demoInfoRepository.save(demoInfo);
    }
    /**
     * 查询数据.
     * @param id
     * @return
     */
    @Cacheable(value=DEMO_CACHE_NAME,key="'demoInfo_'+#id")
    @Override
    public DemoInfo findById(Long id){
        System.err.println("没有走缓存!"+id);
        return demoInfoRepository.findOne(id);
    }
    /**
     * http://www.mincoder.com/article/2096.shtml:
     *
     * 修改数据.
     *
     * 在支持Spring Cache的环境下,对于使用@Cacheable标注的方法,Spring在每次执行前都会检查Cache中是否存在相同key的缓存元素,
     * 如果存在就不再执行该方法,而是直接从缓存中获取结果进行返回,否则才会执行并将返回结果存入指定的缓存中。
     * @CachePut也可以声明一个方法支持缓存功能。与@Cacheable不同的是
     * 使用@CachePut标注的方法在执行前不会去检查缓存中是否存在之前执行过的结果,
     * 而是每次都会执行该方法,并将执行结果以键值对的形式存入指定的缓存中。
     @CachePut也可以标注在类上和方法上。使用@CachePut时我们可以指定的属性跟@Cacheable是一样的。
  *
  * @param updated
 * @return
 *
 * @throws NotFoundException
 */
@CachePut(value = DEMO_CACHE_NAME,key = "'demoInfo_'+#updated.getId()")
//@CacheEvict(value = DEMO_CACHE_NAME,key = "'demoInfo_'+#updated.getId()")//这是清除缓存.
@Override
public DemoInfo update(DemoInfo updated) throws NotFoundException{
    DemoInfo demoInfo = demoInfoRepository.findOne(updated.getId());
    if(demoInfo == null){
        throw new NotFoundException("No find");
    }
    demoInfo.setName(updated.getName());
    demoInfo.setPwd(updated.getPwd());
    return demoInfo;
}
/**
 * 删除数据.
 * @param id
 */
@CacheEvict(value = DEMO_CACHE_NAME,key = "'demoInfo_'+#id")//这是清除缓存.
@Override
public void delete(Long id){
    demoInfoRepository.delete(id);
}

}

编写Controller

@RestController
public class DemoInfoController {

@Resource
private DemoInfoService demoInfoService;

@RequestMapping("/test")
public String test(){
    //存入两条数据.
    DemoInfo demoInfo = new DemoInfo();
    demoInfo.setName("张三");
    demoInfo.setPwd("123456");
    DemoInfo demoInfo2 = demoInfoService.save(demoInfo);
    //不走缓存.
    System.out.println(demoInfoService.findById(demoInfo2.getId()));
    //走缓存.
    System.out.println(demoInfoService.findById(demoInfo2.getId()));
    demoInfo = new DemoInfo();
    demoInfo.setName("李四");
    demoInfo.setPwd("123456");
    DemoInfo demoInfo3 = demoInfoService.save(demoInfo);
    //不走缓存.
    System.out.println(demoInfoService.findById(demoInfo3.getId()));
    //走缓存.
    System.out.println(demoInfoService.findById(demoInfo3.getId()));
    System.out.println("============修改数据=====================");
    //修改数据.
    DemoInfo updated = new DemoInfo();
    updated.setName("李四-updated");
    updated.setPwd("123456");
    updated.setId(demoInfo3.getId());
    try {
        System.out.println(demoInfoService.update(updated));
    } catch (NotFoundException e) {
        e.printStackTrace();
    }
    //不走缓存.
    System.out.println(demoInfoService.findById(updated.getId()));
    return "ok";
}
}

“`

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值