一、背景

当业务实现上需要用到本地缓存,来解决一些数据量相对较小但是频繁访问数据的场景,可以采用Google的CacheBuilder解决方案。

二、代码实现

1. 首先在maven中引入下面的包

<dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
<version>19.0</version>
</dependency>

2. 代码测试案例

import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache; import java.util.concurrent.TimeUnit; public class LocalCacheTest {
// 测试类
public static void main(String[] args) throws Exception {
CacheService us = new CacheService();
for (int i = 0; i < 6; i++) {
System.out.println(us.getName("1001"));
TimeUnit.SECONDS.sleep(1);
}
} // 实现类
public static class CacheService {
private final LoadingCache<String, String> cache; public CacheService() {
/**
* 创建本地缓存,当本地缓存不命中时,调用load方法,返回结果,再缓存结果, 3秒自动过期
*/
cache = CacheBuilder.newBuilder().expireAfterWrite(3, TimeUnit.SECONDS)
.build(new CacheLoader<String, String>() {
public String load(String id) throws Exception {
System.out.println("load()method invoke, 执行查询数据库, 等其他复杂的逻辑");
TimeUnit.MILLISECONDS.sleep(100);
return "User:" + id;
}
});
} public String getName(String id) throws Exception {
long start = System.currentTimeMillis();
String result = cache.get(id);
System.out.println("查询 "+id +" 耗时:"+ (System.currentTimeMillis()-start) + " ms");
return result;
}
}
}

3. 控制台输出

从控制台输出,可以看出,当本地缓存不命中时,调用load方法,通过数据库查询结果,返回结果,再缓存结果, 耗时较长。如果命中查询速度非常快,可达0ms,3秒自动过期后,重复上述操作。

load()method invoke, 执行查询数据库, 等其他复杂的逻辑
查询 1001 耗时:124 ms
User:1001
查询 1001 耗时:0 ms
User:1001
查询 1001 耗时:0 ms
User:1001
load()method invoke, 执行查询数据库, 等其他复杂的逻辑
查询 1001 耗时:108 ms
User:1001
查询 1001 耗时:0 ms
User:1001
查询 1001 耗时:0 ms
User:1001 Process finished with exit code 0

4. 附工具类

import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder; import java.util.concurrent.TimeUnit; public final class JvmCacheUtil { public static final String JVM_CACHE_SPECIFY_GEO = "cache_specify_geo"; public static final String JVM_CACHE_BUILD_SQL = "cache_build_sql"; public static final int DEFAULT_CAPACITY = 50; public static final long DEFAULT_CACHE_EXP = 12L; public static final long DEFAULT_CACHE_ENTRY_EXP = 900L;

/**
* 12小时后过期
*/
private static Cache<String,
Cache> cacheManager = CacheBuilder.newBuilder()
.maximumSize(DEFAULT_CAPACITY)
.expireAfterAccess(12L, TimeUnit.HOURS)
.expireAfterWrite(12L, TimeUnit.HOURS)
.initialCapacity(10)
.build(); public static Cache getCache(String cacheName) {
return getCache(cacheName, DEFAULT_CAPACITY, DEFAULT_CACHE_ENTRY_EXP, TimeUnit.SECONDS);
} public static Cache getCache(String cacheName, long expire, TimeUnit timeUnit) {
return getCache(cacheName, DEFAULT_CAPACITY, expire, timeUnit);
} public static Cache getCache(String cacheName, int capacity, long expire, TimeUnit timeUnit) {
Cache cache = cacheManager.getIfPresent(cacheName);
if (null == cache) {
cache = CacheBuilder.newBuilder()
.maximumSize(DEFAULT_CAPACITY)
.expireAfterAccess(expire, timeUnit)
.expireAfterWrite(expire, timeUnit)
.initialCapacity(capacity)
.build();
cacheManager.put(cacheName, cache);
}
return cache;
} public static Object get(String cacheName, String key) {
Cache cache = getCache(cacheName);
return cache.getIfPresent(key);
} public static void put(String cacheName, String key, Object val) {
Cache cache = getCache(cacheName);
cache.put(key, val);
} public static void put(String cacheName, String key, Object val, long expire, TimeUnit timeUnit) {
Cache cache = getCache(cacheName, expire, timeUnit);
cache.put(key, val);
}

Java本地缓存解决方案---使用Google的CacheBuilder的更多相关文章

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

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

  2. 本地缓存解决方案-Caffeine Cache

    1.1 关于Caffeine Cache ​ Google Guava Cache是一种非常优秀本地缓存解决方案,提供了基于容量,时间和引用的缓存回收方式.基于容量的方式内部实现采用LRU算法,基于引 ...

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

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

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

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

  5. Guava - LoadingCache实现Java本地缓存

    前言 Guava是Google开源出来的一套工具库.其中提供的cache模块非常方便,是一种与ConcurrentMap相似的缓存Map. 官方地址:https://github.com/google ...

  6. java本地缓存

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

  7. JAVA中使用最广泛的本地缓存?Ehcache的自信从何而来 —— 感受来自Ehcache的强大实力

    大家好,又见面了. 本文是笔者作为掘金技术社区签约作者的身份输出的缓存专栏系列内容,将会通过系列专题,讲清楚缓存的方方面面.如果感兴趣,欢迎关注以获取后续更新. 作为<深入理解缓存原理与实战设计 ...

  8. JAVA中使用最广泛的本地缓存?Ehcache的自信从何而来3 —— 本地缓存变身分布式集群缓存,打破本地缓存天花板

    大家好,又见面了. 本文是笔者作为掘金技术社区签约作者的身份输出的缓存专栏系列内容,将会通过系列专题,讲清楚缓存的方方面面.如果感兴趣,欢迎关注以获取后续更新. 上一篇文章中,我们知晓了如何在项目中通 ...

  9. 重新认识下JVM级别的本地缓存框架Guava Cache——优秀从何而来

    大家好,又见面了. 本文是笔者作为掘金技术社区签约作者的身份输出的缓存专栏系列内容,将会通过系列专题,讲清楚缓存的方方面面.如果感兴趣,欢迎关注以获取后续更新. 不知不觉,这已经是<深入理解缓存 ...

  10. 基于ConcurrentHashMap的本地缓存

    基于ConcurrentHashMap的本地缓存 在系统中,有些数据,数据量小,但是访问十分频繁(例如国家标准行政区域数据),针对这种场景,需要将数据搞到应用的本地缓存中,以提升系统的访问效率,减少无 ...

随机推荐

  1. 一些JVM指令

    1.jps:查看本机java进程信息. 2 .jstack:打印线程的栈信息,制作线程dump文件. 3.jmap:打印内存映射,制作堆dump文件 4.jstat:性能监控工具 5.jhat:内存分 ...

  2. redisTemplate类学习及理解

    List<Object> list = masterRedisTemplate.executePipelined((RedisCallback<Long>) connectio ...

  3. Hadoop完全分布式开发配置流程

    完全分布式开发 整体流程 1.准备3台纯净虚拟机 2.修改每台ip,主机名,主机映射,关闭防火墙 3.安装jdk和hadoop,配置环境变量 4.集群分发脚本编写 5.集群配置 6.ssh免密登录 7 ...

  4. 三、核心实战-服务Service-Ingress

    Service 将一组Pods公开为网络服务的抽象方法. 暴露deployment只能在集群内访问是ClusterIP,可以集群外访问是NodePort,默认端口分配是30000-32767之间 ku ...

  5. Python SQLite创建数据库和数据表及数据的增删改查

    SQLite是一款轻型的SQL类型数据库,处理速度快且占用资源非常低,Python自带, 不需要配置不需要任何外部的依赖.数据库本身就是一个.db文件,非常适合存储本地数据.本文主要介绍Python ...

  6. 杂:python_windows标准输出带颜色

    import ctypes handle = ctypes.windll.kernel32.GetStdHandle(-11) # -11 for output   -10 for input   - ...

  7. 掌控安全学院SQL注入靶场-布尔盲注(三)

    测试了username参数,没有发现注入 123456' or '1'='1 123456' or '1'='2 第二种注入方法

  8. 在wsl 2中编译自己的魔趣(mokee) ROM

    1.安装wsl2 a.在windows 10 系统中启用wsl2(只能是wsl2,wsl1 会编译失败), 并商店中下载ubuntu镜像,商店默认安装位置为C:\Program Files\Windo ...

  9. Elasticsearch 实战

    需求 假设现在有这么一个需求,系统接了很多的报文,需要提供全文检索,为了简化,报文目前只有类型,流水号,内容这三个字段. 索引设计 建立msg索引,映射规则如下 PUT /msg { "ma ...

  10. js中的this的指向问题

    this的指向在函数定义的时候是确定不了的,只有函数执行的时候才能确定this到底指向谁,实际上this的最终指向的是那个调用它的对象 this永远指向的是最后调用它的对象,也就是看它执行的时候是谁调 ...