前言

Guava是Google开源出来的一套工具库。其中提供的cache模块非常方便,是一种与ConcurrentMap相似的缓存Map。

官方地址:https://github.com/google/guava/wiki/CachesExplained

开始构建

一. 添加依赖

<dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
<version>27.1-jre</version>
</dependency>

二.创建 CacheLoader

 1 LoadingCache<Long, String> cache = CacheBuilder.newBuilder()
2 //缓存池大小,在缓存项接近该大小时, Guava开始回收旧的缓存项
3 .maximumSize(GUAVA_CACHE_SIZE)
4 //设置时间对象没有被读/写访问则对象从内存中删除(在另外的线程里面不定期维护)
5 .expireAfterAccess(10, TimeUnit.MINUTES)
6 //移除监听器,缓存项被移除时会触发
7 .removalListener(new RemovalListener <Long, String>() {
8 @Override
9 public void onRemoval(RemovalNotification<Long, String> rn) {
10 //执行逻辑操作
11 }
12 })
13 //开启Guava Cache的统计功能
14 .recordStats()
15 .build(cacheLoader);

三.个人封装的工具类

  1 package com.xxx;
2
3 import com.google.common.cache.*;
4 import org.slf4j.Logger;
5
6 import java.util.ArrayList;
7 import java.util.List;
8 import java.util.Map;
9 import java.util.concurrent.ConcurrentMap;
10 import java.util.concurrent.TimeUnit;
11
12 public class CacheManager {
13
14 private static Logger log = Log.get();
15
16 /** 缓存项最大数量 */
17 private static final long GUAVA_CACHE_SIZE = 100000;
18
19 /** 缓存时间:天 */
20 private static final long GUAVA_CACHE_DAY = 10;
21
22 /** 缓存操作对象 */
23 private static LoadingCache<Long, String> GLOBAL_CACHE = null;
24
25 static {
26 try {
27 GLOBAL_CACHE = loadCache(new CacheLoader <Long, String>() {
28 @Override
29 public String load(Long key) throws Exception {
30 // 处理缓存键不存在缓存值时的处理逻辑
31 return "";
32 }
33 });
34 } catch (Exception e) {
35 log.error("初始化Guava Cache出错", e);
36 }
37 }
38
39 /**
40 * 全局缓存设置
41 *
42 * 缓存项最大数量:100000
43 * 缓存有效时间(天):10
44 *
45 *
46 * @param cacheLoader
47 * @return
48 * @throws Exception
49 */
50 private static LoadingCache<Long, String> loadCache(CacheLoader<Long, String> cacheLoader) throws Exception {
51 LoadingCache<Long, String> cache = CacheBuilder.newBuilder()
52 //缓存池大小,在缓存项接近该大小时, Guava开始回收旧的缓存项
53 .maximumSize(GUAVA_CACHE_SIZE)
54 //设置时间对象没有被读/写访问则对象从内存中删除(在另外的线程里面不定期维护)
55 .expireAfterAccess(GUAVA_CACHE_DAY, TimeUnit.DAYS)
56 // 设置缓存在写入之后 设定时间 后失效
57 .expireAfterWrite(GUAVA_CACHE_DAY, TimeUnit.DAYS)
58 //移除监听器,缓存项被移除时会触发
59 .removalListener(new RemovalListener <Long, String>() {
60 @Override
61 public void onRemoval(RemovalNotification<Long, String> rn) {
62 //逻辑操作
63 }
64 })
65 //开启Guava Cache的统计功能
66 .recordStats()
67 .build(cacheLoader);
68 return cache;
69 }
70
71 /**
72 * 设置缓存值
73 * 注: 若已有该key值,则会先移除(会触发removalListener移除监听器),再添加
74 *
75 * @param key
76 * @param value
77 */
78 public static void put(Long key, String value) {
79 try {
80 GLOBAL_CACHE.put(key, value);
81 } catch (Exception e) {
82 log.error("设置缓存值出错", e);
83 }
84 }
85
86 /**
87 * 批量设置缓存值
88 *
89 * @param map
90 */
91 public static void putAll(Map<? extends Long, ? extends String> map) {
92 try {
93 GLOBAL_CACHE.putAll(map);
94 } catch (Exception e) {
95 log.error("批量设置缓存值出错", e);
96 }
97 }
98
99 /**
100 * 获取缓存值
101 * 注:如果键不存在值,将调用CacheLoader的load方法加载新值到该键中
102 *
103 * @param key
104 * @return
105 */
106 public static String get(Long key) {
107 String token = "";
108 try {
109 token = GLOBAL_CACHE.get(key);
110 } catch (Exception e) {
111 log.error("获取缓存值出错", e);
112 }
113 return token;
114 }
115
116 /**
117 * 移除缓存
118 *
119 * @param key
120 */
121 public static void remove(Long key) {
122 try {
123 GLOBAL_CACHE.invalidate(key);
124 } catch (Exception e) {
125 log.error("移除缓存出错", e);
126 }
127 }
128
129 /**
130 * 批量移除缓存
131 *
132 * @param keys
133 */
134 public static void removeAll(Iterable<Long> keys) {
135 try {
136 GLOBAL_CACHE.invalidateAll(keys);
137 } catch (Exception e) {
138 log.error("批量移除缓存出错", e);
139 }
140 }
141
142 /**
143 * 清空所有缓存
144 */
145 public static void removeAll() {
146 try {
147 GLOBAL_CACHE.invalidateAll();
148 } catch (Exception e) {
149 log.error("清空所有缓存出错", e);
150 }
151 }
152
153 /**
154 * 获取缓存项数量
155 *
156 * @return
157 */
158 public static long size() {
159 long size = 0;
160 try {
161 size = GLOBAL_CACHE.size();
162 } catch (Exception e) {
163 log.error("获取缓存项数量出错", e);
164 }
165 return size;
166 }
167 }

总结

1.移除机制

guava做cache时候数据的移除分为被动移除主动移除两种。

被动移除分为三种:1).基于大小的移除:数量达到指定大小,会把不常用的键值移除

         2).基于时间的移除:expireAfterAccess(long, TimeUnit) 根据某个键值对最后一次访问之后多少时间后移除 
                   expireAfterWrite(long, TimeUnit) 根据某个键值对被创建或值被替换后多少时间移除

            3).基于引用的移除:主要是基于java的垃圾回收机制,根据键或者值的引用关系决定移除

主动移除分为三种:1).单独移除:Cache.invalidate(key)

         2).批量移除:Cache.invalidateAll(keys)

         3).移除所有:Cache.invalidateAll()

如果配置了移除监听器RemovalListener,则在所有移除的动作时会同步执行该listener下的逻辑。

如需改成异步,使用:RemovalListeners.asynchronous(RemovalListener, Executor)

2.遇到的问题

1). 在put操作之前,如果已经有该键值,会先触发removalListener移除监听器,再添加

2). 配置了expireAfterAccess和expireAfterWrite,但在指定时间后没有被移除。

 解决方案:CacheBuilder在文档上有说明:If expireAfterWrite or expireAfterAccess is requested entries may be evicted on each cache modification, on occasional cache accesses, or on calls to Cache.cleanUp(). Expired entries may be counted in Cache.size(), but will never be visible to read or write operations. 翻译过来大概的意思是:CacheBuilder构建的缓存不会在特定时间自动执行清理和回收工作,也不会在某个缓存项过期后马上清理,它不会启动一个线程来进行缓存维护,因为a)线程相对较重,b)某些环境限制线程的创建。它会在写操作时顺带做少量的维护工作,或者偶尔在读操作时做。当然,也可以创建自己的维护线程,以固定的时间间隔调用Cache.cleanUp()。

Guava - LoadingCache实现Java本地缓存的更多相关文章

  1. Java本地缓存解决方案其一(使用Google的CacheBuilder)

    前不久,业务实现上需要用到本地缓存来解决一些数据量相对较小但是频繁访问的数据,通过查找各种资料,找到了一种可以实现的方案--采用的是Google的CacheBuilder.下面是代码实现过程:1.首先 ...

  2. Caffeine Cache-高性能Java本地缓存组件

    前面刚说到Guava Cache,他的优点是封装了get,put操作:提供线程安全的缓存操作:提供过期策略:提供回收策略:缓存监控.当缓存的数据超过最大值时,使用LRU算法替换.这一篇我们将要谈到一个 ...

  3. Guava的两种本地缓存策略

    Guava的两种缓存策略 缓存在很多场景下都需要使用,如果电商网站的商品类别的查询,订单查询,用户基本信息的查询等等,针对这种读多写少的业务,都可以考虑使用到缓存.在一般的缓存系统中,除了分布式缓存, ...

  4. 实现 Java 本地缓存,该从这几点开始

    缓存,我相信大家对它一定不陌生,在项目中,缓存肯定是必不可少的.市面上有非常多的缓存工具,比如 Redis.Guava Cache 或者 EHcache.对于这些工具,我想大家肯定都非常熟悉,所以今天 ...

  5. Guava Cache,Java本地内存缓存使用实践

    Guava Cache,网上介绍很多,我就不赘述了. 分享一篇好的文章: Guava Cache内存缓存使用实践-定时异步刷新及简单抽象封装 Google Guava 3-缓存 在原作者基础上,我做了 ...

  6. java本地缓存

    1.为什么要使用缓存 由于服务器.数据库.网络等资源有限,无法支撑越来越多的请求与计算量,所以将一部分数据放在缓存中,以此减小薄弱环节的计算量和请求流程. 网站中缓存的应用场景:        1:可 ...

  7. springboot之本地缓存(guava与caffeine)

    1. 场景描述 因项目要使用本地缓存,具体为啥不用redis等,就不讨论,记录下过程,希望能帮到需要的朋友. 2.解决方案 2.1 使用google的guava作为本地缓存 初步的想法是使用googl ...

  8. 使用guava cache在本地缓存热点数据

    某些热点数据在短时间内可能会被成千上万次访问,所以除了放在redis之外,还可以放在本地内存,也就是JVM的内存中. 我们可以使用google的guava cache组件实现本地缓存,之所以选择gua ...

  9. 第七章 企业项目开发--本地缓存guava cache

    1.在实际项目开发中,会使用到很多缓存技术,而且数据库的设计一般也会依赖于有缓存的情况下设计. 常用的缓存分两种:本地缓存和分布式缓存. 常用的本地缓存是guava cache,本章主要介绍guava ...

随机推荐

  1. Python开发桌面微型计算器

    开发Windows窗口需要用到tkinter库 所以上来的第一件事就是: import tkinter as t window = t.Tk()#创建了一个窗口 window.title('微型计算器 ...

  2. Redis挖矿原理及防范

    笔者也曾经被挖矿病毒侵袭过,灰常难受,但是其实你只要了解入侵的手段就非常好防范了,今天我们就演示一下如果通过Redis进行提权获取远程服务器的Root用户. 1.首先我们需要一些先决条件 条件一:你首 ...

  3. 若依管理系统RuoYi-Vue(一):项目启动和菜单创建

    若依管理系统应该是国内最受欢迎的完全开源的后端管理系统了吧,看看gitee上的star数量,着实惊人.若依系统有很多个版本 版本 gitee地址 说明 前后端不分离版本 https://gitee.c ...

  4. 2019 Multi-University Training Contest 1 D.Vacation(思维)

    题意:有n俩车行驶在一条道路上,每辆车有车长li,距离终点的距离si,速度vi,不能超出,并且驶过终点后会依旧保持原状态行驶,问最后一辆车过终点的时间. 思路:因为行驶过终点后还是要保持之前的行驶方式 ...

  5. 组合数取模及Lucas定理

    引入: 组合数C(m,n)表示在m个不同的元素中取出n个元素(不要求有序),产生的方案数.定义式:C(m,n)=m!/(n!*(m-n)!)(并不会使用LaTex QAQ). 根据题目中对组合数的需要 ...

  6. Bézout恒等式

    写在前面: 记录了个人的学习过程,同时方便复习 整理自网络 非原创部分会标明出处 目录 结论 证明 拓展 n个整数间 拓展欧几里得算法 拓展欧几里得算法的多解 结论 (Bézout / 裴蜀 / 贝祖 ...

  7. LianLianKan HDU - 4272 状压dp

    题意:长度为n(n<=1000)的栈,栈顶元素可以与下面1~5个数中相同的元素消去,问最后能都完全消去. 题解: 比如这个序列12345678910112这个位置的最远可匹配位置能到11为什么呢 ...

  8. 服务注册与发现-Eureka (高可用设计)

    什么是高可用 部署需要考虑的是什么: 1.系统遇到单点失效问题,如何能够快速切换到其他节点完成任务 2.如何应对网络故障,即系统如何设计成"故障开放型"(expecting fai ...

  9. woj1009 最短路 The Legend of Valiant Emigration

    title: woj1009 最短路 The Legend of Valiant Emigration date: 2020-03-07 categories: acm tags: [acm,最短路, ...

  10. LeetCode 856. Score of Parentheses 括号的分数

    其实是这道题的变式(某港带同学的C/C++作业) 增加一点难度,输入的S不一定为平衡的,需要自己判断是否平衡,若不平衡输出为0. 题目描述 Given a parentheses string s, ...