缓存

缓存分为本地缓存与分布式缓存。本地缓存为了保证线程安全问题,一般使用ConcurrentMap的方式保存在内存之中,而常见的分布式缓存则有RedisMongoDB等。

  • 一致性:本地缓存由于数据存储于内存之中,每个实例都有自己的副本,可能会存在不一致的情况;分布式缓存则可有效避免这种情况
  • 开销:本地缓存会占用JVM内存,会影响GC及系统性能;分布式缓存的开销则在于网络时延和对象序列化,故主要影响调用时延
  • 适用场景:本地缓存适用于数据量较小或变动较少的数据;分布式缓存则适用于一致性要求较高及数量量大的场景(可弹性扩容)

本地缓存适用于数据量较小或变动较少的数据,因为变动多需要考虑到不同实例的缓存一致性问题,而数据量大则需要考虑缓存回收策略及GC相关的问题

Guava cache

Guava Cache 是Google Fuava中的一个内存缓存模块,用于将数据缓存到JVM内存中。

  • 提供了get、put封装操作,能够集成数据源 ;
  • 线程安全的缓存,与ConcurrentMap相似,但前者增加了更多的元素失效策略,后者只能显示的移除元素;
  • Guava Cache提供了多种基本的缓存回收方式
  • 监控缓存加载/命中情况

通常,Guava缓存适用于以下情况:

  • 愿意花费一些内存来提高速度。
  • 使用场景有时会多次查询key。
  • 缓存将不需要存储超出RAM容量的数据

详细配置

缓存的并发级别

Guava提供了设置并发级别的API,使得缓存支持并发的写入和读取。与ConcurrentHashMap类似,Guava cache的并发也是通过分离锁实现。在通常情况下,推荐将并发级别设置为服务器cpu核心数。

CacheBuilder.newBuilder()// 设置并发级别为cpu核心数,默认为4.concurrencyLevel(Runtime.getRuntime().availableProcessors()) .build();

缓存的初始容量设置

我们在构建缓存时可以为缓存设置一个合理大小初始容量,由于Guava的缓存使用了分离锁的机制,扩容的代价非常昂贵。所以合理的初始容量能够减少缓存容器的扩容次数。

CacheBuilder.newBuilder()// 设置初始容量为100.initialCapacity(100).build();

设置最大存储

Guava Cache可以在构建缓存对象时指定缓存所能够存储的最大记录数量。当Cache中的记录数量达到最大值后再调用put方法向其中添加对象,Guava会先从当前缓存的对象记录中选择一条删除掉,腾出空间后再将新的对象存储到Cache中。

CacheBuilder.newBuilder()// 设置最大容量为1000.maximumSize(1000).build();

缓存清除策略

  1. 基于存活时间的清除策略
  • expireAfterWrite 写缓存后多久过期
  • expireAfterAccess 读写缓存后多久过期
    存活时间策略可以单独设置或组合配置
  1. 基于容量的清除策略
    通过CacheBuilder.maximumSize(long)方法可以设置Cache的最大容量数,当缓存数量达到或接近该最大值时,Cache将清除掉那些最近最少使用的缓存

  2. 基于权重的清除 策略
    使用CacheBuilder.weigher(Weigher)指定一个权重函数,并且用CacheBuilder.maximumWeight(long)指定最大总重。

如每一项缓存所占据的内存空间大小都不一样,可以看作它们有不同的“权重”(weights),作为执行清除策略时优化回收的对象

LoadingCache<Key, Graph> graphs = CacheBuilder.newBuilder().maximumWeight(100000).weigher(new Weigher<Key, Graph>() {public int weigh(Key k, Graph g) {return g.vertices().size();}}).build(new CacheLoader<Key, Graph>() {public Graph load(Key key) { // no checked exceptionreturn createExpensiveGraph(key);}});
  1. 显式清除
  • 清除单个key:Cache.invalidate(key)
  • 批量清除key:Cache.invalidateAll(keys)
  • 清除所有缓存项:Cache.invalidateAll()
  1. 基于引用的清除策略
    在构建Cache实例过程中,通过设置使用弱引用的键、或弱引用的值、或软引用的值,从而使JVM在GC时顺带实现缓存的清除
  • CacheBuilder.weakKeys():使用弱引用存储键。当键没有其它(强或软)引用时,缓存项可以被垃圾回收
  • CacheBuilder.weakValues():使用弱引用存储值。当值没有其它(强或软)引用时,缓存项可以被垃圾回收
  • CacheBuilder.softValues():使用软引用存储值。软引用只有在响应内存需要时,才按照全局最近最少使用的顺序回收。考虑到使用软引用的性能影响,我们通常建议使用更有性能预测性的缓存大小限定

垃圾回收仅依赖==恒等式,使用弱引用键的缓存用而不是equals(),即同一对象引用。

Cache

显式put操作置入内存

private static Cache<Integer, Integer> numCache = CacheBuilder.newBuilder().expireAfterWrite(5, TimeUnit.MINUTES).build();public static void main(String[] args) throws Exception {System.out.println(numCache.getIfPresent(1));Thread.sleep(1000);System.out.println(numCache.getIfPresent(1));Thread.sleep(1000);numCache.put(1, 5);System.out.println(numCache.getIfPresent(1));// console: null null 5
}

LoadingCache

使用自定义ClassLoader加载数据,置入内存中。从LoadingCache中获取数据时,若数据存在则直接返回;若数据不存在,则根据ClassLoaderload方法加载数据至内存,然后返回该数据

private static LoadingCache<Integer,Integer> numCache = CacheBuilder.newBuilder().expireAfterWrite(5L, TimeUnit.MINUTES).maximumSize(5000L).build(new CacheLoader<Integer, Integer>() {@Overridepublic Integer load(Integer key) throws Exception {System.out.println("no cache");return key * 5;}});public static void main(String[] args) throws Exception {System.out.println(numCache.get(1));Thread.sleep(1000);System.out.println(numCache.get(1));Thread.sleep(1000);numCache.put(1, 6);System.out.println(numCache.get(1));// console: 5 5 6
}

参考资料:

  1. Google CachesExplained

Guava缓存详解及使用相关推荐

  1. [面试专题]Web缓存详解

    Web缓存详解 标签(空格分隔): 缓存 缓存之于性能优化 请求更快:通过将内容缓存在本地浏览器或距离最近的缓存服务器(如CDN),在不影响网站交互的前提下可以大大加快网站加载速度. 降低服务器压力: ...

  2. MyBatis 一级缓存二级缓存详解

    相关内容: 架构师系列内容:架构师学习笔记(持续更新) MyBatis 缓存详解 cache 缓存 缓存是一般的ORM 框架都会提供的功能,目的就是提升查询的效率和减少数据库的压力.跟Hibernat ...

  3. Hibernate二级缓存详解(转)

    Hibernate二级缓存详解(转) 本文转载 http://www.blogjava.net/supercrsky/articles/238580.html 与Session相对的是,Session ...

  4. vue里页面的缓存详解

    关于vue里页面的缓存详解 实用的例子ABC 往下查看 keep-alive是vue内置的一个组件,可以使被它包含的组件处于保留状态,或避免被重新渲染. 用法: 在keep-alive标签内部添加 i ...

  5. Spring循环依赖和三级缓存详解

    Spring循环依赖和三级缓存详解 Spring在启动过程中,使用到了三个map,称为三级缓存 我们可以这样理解,假设,我们只有一个缓存容器,并且缓存是直接开放给用户可以调用的,如果将未完成赋值的Be ...

  6. cpu二级缓存和一级缓存详解及区别(图解)

    cpu二级缓存和一级缓存详解及区别(图解) 2012-09-02 12:27:55|  分类: 硬件技术 |字号 订阅 处理器缓存的传输速率确实很高,然而还不足以取代内存的地位,这主要是由于缓存只是内 ...

  7. Mybatis二级缓存详解

    Mybatis二级缓存 Mybatis相关全览 一.简介 二.一级缓存 1.入口 2.演示 案例一: 案例二: 3.总结 三.二级缓存 1.入口 2.如何开启二级缓存 cache-ref配置 cach ...

  8. MyBatis 缓存详解-开启二级缓存的方法

    第一步:在mybatis-config.xml 中配置了(可以不配置,默认是true): <setting name="cacheEnabled" value="t ...

  9. HTTP缓存详解之etag

    2019独角兽企业重金招聘Python工程师标准>>> ----------------- Apache 1.3和2.x中的ETag格式为inode-size-timestamp.即 ...

最新文章

  1. 【Qt】Qt发布可执行程序(打包依赖库)
  2. Python多任务(多线程执行带有参数的任务,利用threading创建线程时传入参数--args参数和kwargs参数)
  3. Android开发;Activity-Hook你了解多少?一起来debug
  4. 受检异常和非受检异常
  5. JavaScript中Element与Node的区别,children与childNodes的区别
  6. Spring :读取配置文件(.properties、.yam)相关注解
  7. 057 Java中kafka的Producer程序实现
  8. Python-初体验
  9. bzoj 3750: [POI2015]Pieczęć(模拟)
  10. IE开发者人员工具使用说明 pptx
  11. 后台管理页面(bootdo)
  12. Habor数据迁移方式有多少,skopeo效率最好
  13. win10和win server 2016新的TCP功能介绍
  14. Windows Live Writer插件:在WLW中插入语法高亮代码
  15. 硬币面值组合问题(完全背包问题)
  16. HTML、CSS定义字体、颜色、背景等属性
  17. Java变成笔记4:复用类
  18. NC207040 丢手绢
  19. 《你好哇,程序员》新鲜出炉
  20. 笔记本键盘进水和键帽问题解决

热门文章

  1. 量化投资学习-13:一张图残酷的展现了庄家、量化交易者、散户的盈利空间的对比
  2. 人民日报“点名”绿之韵,大数据助力健康产业升级新引擎
  3. NVIDIA Drive PX2 使用笔记
  4. Web安全漏洞笔记整合
  5. Qt——音乐播放控件
  6. 关于如何插入网线(网线没网)的同时连接wifi上网
  7. 【装13手册】MAC常用终端命令及快捷键操作
  8. Cyanine3-PEG-PEI/Cholesterol,花菁染料PEG2K聚合物/胆固醇
  9. 【配电网重构】基于粒子群求解配电网重构模型matlab源码
  10. 图片与文字间出现间隙的原因与解决方案