• 缓存

    • 服务器自身(内存)的缓存

      • 利用java程序中的变量

        • 简单
        • 集群环境中多个实例无法共享同步
    • 缓存服务器(一般支持集群、分布式)
      • Redis
      • Memcached
  • Spring中使用注解使用缓存
    • 启动类或者配置类

      • 使用@EnableCaching,必须使用,否则后续注解虽然不报错,但没有使用缓存
      • 可以放一个@Bean,用来注入CacheManager
@EnableCaching
@EnableScheduling
@Configuration
public class CacheConfig {

    @Bean
    @Primary
    public CacheManager cacheManager() {
        return new ConcurrentMapCacheManager("assets");
    }

}
* 使用注解标识方法()
    * @Cacheable,通常用在get方法上
    * @CachePut:运行后会更新/新建缓存中的值,通常用在update方法上
    * @CacheEvict:运行后会删掉缓存中的值,通常用在delete方法上
* 使用注解标识类(方法上就不用标识了)
    * @CacheConfig:相当于该类下所有的方法的cacheNames都是其中指定的字符串?
* 参数
    * 标识缓存项,都是用注解中的cacheNames参数和key参数组合起来的:**缓存的key是cacheNames::key**
        * cacheNames可以是一个字符串数组,可以对应多个缓存的key,每个方法都需要指定cacheNames参数
        * key
            * 通常用SpringEL表达式。
            * 以\#开头,内部以.进行分级分隔。
            * **默认按照方法的参数自动生成**
            * 可以通过keyGenerator参数自定义生成方式
            * 有时key可以省略,key有默认的机制。如果方法只有一个参数,那么就会用cacheNames和参数值组合成key;如果有多个参数的时候,就会把多个参数组合起来再组合成key。**所有要看情况,key有时可以省略,有时不可以。**
    * condition可以指定一个条件,用于执行方法前判断是否使用注解的功能
    * unless用于执行方法后,用返回值?判断是否使用注解提供的功能
    * cacheManager、CacheResolver、keyGenerator
    * @CacheEvict多了两个参数
        * allEntries:值为true则不管key,清除cacheName下所有key的缓存
        * beforeInvocation:值为true,执行方法前清除,默认是执行方法后清除
* 设置缓存过期
    * 可以和Scheduled Tasks结合,使用CacheEvict注解定时清除缓存
    * 某些缓存服务如Redis,可以通过application.yml配置一个全局的过期时间(默认不过期)
  • 使用Redis

    • 在maven或者gradle中加入依赖
    • 在application.yml中加入配置
    • 可以通过使用RedisTemplate(RabbitMQ中有类似的概念用来发数据到队列)单独设置一些过期时间等
    • 有可能的坑
      • application.yml中其实可以配置key前缀、是否使用key前缀。如果打开了这两个开关,那么有可能导致代码中的cacheNames参数失效,如果两个方法的key相同,那么会导致缓存的key相同导致混乱。根本原因是spring把cacheNames参数也当成了application.yml中所谓的key,所以这两个参数一般不要配
    • 使用
      • 需要在build.gradle中添加依赖:compile('org.springframework.boot:spring-boot-starter-data-redis'),然后就可以使用RedisTemplate<String, XXX>类了
      • application.properties文件中的配置
      • 配置类(其实这一个就够了)
        • 注入生成的redisSetRepository可以直接用来进行redis操作,比如redisRepository.put
@Configuration
public class RedisConfig {

    @Autowired
    private RedisConnectionFactory redisConnectionFactory;

    @Value("${oee.cache.ttl.hours:24}")
    private long timeout;

    @Bean
    public RedisRepository redisRepository()
    {
        return new RedisRepository(cacheEntryRedisTemplate(), timeout, TimeUnit.HOURS);
    }

    @Bean
    public RedisSetRepository redisSetRepository()
    {
        return new RedisSetRepository(stringRedisTemplate());
    }

    private RedisTemplate<String, CacheEntry> cacheEntryRedisTemplate()
    {
        StringRedisSerializer stringSerializer = new StringRedisSerializer(StandardCharsets.UTF_8);
        Jackson2JsonRedisSerializer<CacheEntry> contentSerializer = new Jackson2JsonRedisSerializer<>(CacheEntry.class);
        RedisTemplate<String, CacheEntry> template = new RedisTemplate<>();
        template.setConnectionFactory(redisConnectionFactory);
        template.setKeySerializer(stringSerializer);
        template.setHashKeySerializer(stringSerializer);
        template.setHashValueSerializer(contentSerializer);
        template.setValueSerializer(contentSerializer);
        template.setEnableDefaultSerializer(false);
        template.afterPropertiesSet();
        return template;
    }

    private RedisTemplate<String, String> stringRedisTemplate()
    {
        StringRedisSerializer stringSerializer = new StringRedisSerializer(StandardCharsets.UTF_8);
        Jackson2JsonRedisSerializer<String> contentSerializer = new Jackson2JsonRedisSerializer<>(String.class);
        RedisTemplate<String, String> template = new RedisTemplate<>();
        template.setConnectionFactory(redisConnectionFactory);
        template.setKeySerializer(stringSerializer);
        template.setHashKeySerializer(stringSerializer);
        template.setHashValueSerializer(contentSerializer);
        template.setValueSerializer(contentSerializer);
        template.setEnableDefaultSerializer(false);
        template.afterPropertiesSet();
        return template;
    }
}
    * 也可以进一步封装一个操作类OeeResourceCache,和一个数据类CacheEntry(包含创建时间)作为value放到redis中
@Configuration
public class OeeResourceCacheConfig {

    @Autowired
    private RedisRepository redisRepository;

    @Autowired
    private RedisSetRepository redisSetRepository;

    @Bean
    public OeeResourceCache oeeResourceCache() {
        return new OeeResourceCache(redisRepository);
    }

    @Bean
    public SynchronizationCache synchronizationCache() {
        return new SynchronizationCache(redisSetRepository);
    }
}

public class OeeResourceCache {

    private static final Logger LOG = LoggerFactory.getLogger(OeeResourceCache.class);

    private static final String OEE_RESOURCE_KEY = "oeeResource";

    private RedisRepository redisRepository;

    public OeeResourceCache(RedisRepository redisRepository) {
        this.redisRepository = redisRepository;
    }

    public void cacheOeeResource(String tenantId, String assetId, OeeResource oeeResource) {

        LOG.debug("Caching OeeResource for tenant {}, assetId {}", tenantId, assetId);
        LOG.trace("OeeResource: {}", oeeResource);

        Check.notNull(tenantId, "tenantId");
        Check.notNull(assetId, "assetId");

        String key = getKey(tenantId, assetId);

        CacheEntry cacheEntry = redisRepository.put(key, CacheEntry.from(oeeResource.toTimeSeriesValue()));

        LOG.info("OeeResource is cached for tenant {}, assetId {}", tenantId, assetId);
        LOG.trace("CacheEntry: {}", cacheEntry);
    }

    public OeeResource getOeeResource(String tenantId, String assetId) {

        LOG.debug("Getting OeeResource from cache for tenant {}, assetId {}", tenantId, assetId);

        Check.notNull(tenantId, "tenantId");
        Check.notNull(assetId, "assetId");

        String key = getKey(tenantId, assetId);

        CacheEntry cacheEntry = redisRepository.get(key);

        LOG.trace("CacheEntry: {}", cacheEntry);

        return cacheEntry == null ? null : OeeResource.from(cacheEntry.to(TimeSeriesValue.class));
    }

    public Optional<OeeResource> getOeeResource(String tenantId, String assetId, Instant time) {
        return Optional.ofNullable(getOeeResource(tenantId, assetId)).filter(Objects::nonNull)
                .filter(oeeResource -> oeeResource.getDate().equals(time));
    }

    private String getKey(String tenantId, String assetId) {
        return String.join(":", OEE_RESOURCE_KEY, tenantId, assetId);
    }
}

@Getter
@Setter
@ToString
public class CacheEntry {

    private String data;

    @JsonInstant
    private Instant createdAt;

    public static CacheEntry from(Object object) {
        CacheEntry entry = new CacheEntry();
        entry.setCreatedAt(Instant.now());
        entry.setData(Misc.serializeSafeJson(object));
        return entry;
    }

    public <T> T to(Class<T> theClass) {
        return Misc.deserializeSafeJson(data, theClass);
    }
}

Spring Boot - Spring Cache的更多相关文章

  1. spring boot guava cache 缓存学习

    http://blog.csdn.net/hy245120020/article/details/78065676 ****************************************** ...

  2. spring Boot+spring Cloud实现微服务详细教程第二篇

    上一篇文章已经说明了一下,关于spring boot创建maven项目的简单步骤,相信很多熟悉Maven+Eclipse作为开发常用工具的朋友们都一目了然,这篇文章主要讲解一下,构建spring bo ...

  3. spring Boot+spring Cloud实现微服务详细教程第一篇

    前些天项目组的大佬跟我聊,说项目组想从之前的架构上剥离出来公用的模块做微服务的开发,恰好去年的5/6月份在上家公司学习了国内开源的dubbo+zookeeper实现的微服务的架构.自己平时对微服务的设 ...

  4. Spring boot +Spring Security + Thymeleaf 认证失败返回错误信息

    [Please make sure to select the branch corresponding to the version of Thymeleaf you are using] Stat ...

  5. 255.Spring Boot+Spring Security:使用md5加密

    说明 (1)JDK版本:1.8 (2)Spring Boot 2.0.6 (3)Spring Security 5.0.9 (4)Spring Data JPA 2.0.11.RELEASE (5)h ...

  6. 256.Spring Boot+Spring Security: MD5是加密算法吗?

    说明 (1)JDK版本:1.8 (2)Spring Boot 2.0.6 (3)Spring Security 5.0.9 (4)Spring Data JPA 2.0.11.RELEASE (5)h ...

  7. Spring Boot+Spring Security:获取用户信息和session并发控制

    说明 (1)JDK版本:1.8(2)Spring Boot 2.0.6(3)Spring Security 5.0.9(4)Spring Data JPA 2.0.11.RELEASE(5)hiber ...

  8. Cola Cloud 基于 Spring Boot, Spring Cloud 构建微服务架构企业级开发平台

    Cola Cloud 基于 Spring Boot, Spring Cloud 构建微服务架构企业级开发平台: https://gitee.com/leecho/cola-cloud

  9. spring boot + spring batch 读数据库文件写入文本文件&读文本文件写入数据库

    好久没有写博客,换了一家新公司,原来的公司用的是spring,现在这家公司用的是spring boot.然后,项目组布置了一个任务,关于两个数据库之间的表同步,我首先想到的就是spring batch ...

  10. Spring Boot/Spring Cloud、ESB、Dubbo

    如何使用Spring Boot/Spring Cloud 实现微服务应用spring Cloud是一个基于Spring Boot实现的云应用开发工具,它为基于JVM的云应用开发中的配置管理.服务发现. ...

随机推荐

  1. Ubuntu更新命令 <转>

    apt-cache search package 搜索包 apt-cache show package 获取包的相关信息,如说明.大小.版本等 sudo apt-get install package ...

  2. 整理SSH框架的优缺点

    Hibernate优点(1) 对象/关系数据库映射(ORM)它使用时只需要操纵对象,使开发更对象化,抛弃了数据库中心的思想,完全的面向对象思想(2) 透明持久化(persistent)带有持久化状态的 ...

  3. TrinityCore3.3.5编译过程-官方指导-踩坑总结

    官方指导:主页->how to compile -> windows 指导文档写得很详细,但有不少细节点没提到,这里把过程简化总结,说明重点,及易坑点 1,安装需求 编译工具:cmake, ...

  4. java中执行子类的构造方法时,会不会先执行父类的构造方法

    会,在创建子类的对象时,jvm会首先执行父类的构造方法,然后再执行子类的构造方法,如果是多级继承,会先执行最顶级父类的构造方法,然后依次执行各级个子类的构造方法.

  5. tomcat https

    转自 http://11lingxian.iteye.com/blog/1491607 双向认证: 客户端向服务器发送消息,首先把消息用客户端证书加密然后连同时把客户端证书一起发送到服务器端, 服务器 ...

  6. 88. Merge Sorted Array (Array)

    Given two sorted integer arrays nums1 and nums2, merge nums2 into nums1 as one sorted array. Note: Y ...

  7. linux ifconfig显示 command not found

    本人装的是centos7 想看下网络配置 结果显示如图: 正常情况下 ifconfig  是在超级管理员 的所属的目录 sbin/下的命令 现在来查看该目录下. 没有找到,别急 用 yum  sear ...

  8. fastdfs 有用 新增tracker或storage

    FastDFS是一个开源的轻量级分布式文件系统,它对文件进行管理,功能包括:文件存储.文件同步.文件访问(文件上传.文件下载)等,解决了大容量存储和负载均衡的问题.特别适合以文件为载体的在线服务,如相 ...

  9. CentOS安装配置Tomcat-7

    安装环境:CentOS-6.5安装方式:源码安装软件:apache-tomcat-7.0.29.tar.gz下载地址:http://tomcat.apache.org/download-70.cgi ...

  10. 面向对象的JavaScript-009-闭包

    引自:https://developer.mozilla.org/cn/docs/Web/JavaScript/Closures 闭包是指能够访问自由变量的函数 (变量在本地使用,但在闭包中定义).换 ...