A comparison of local caches (1) 【本地缓存之比较 (1)】
1. Spring local cache 【Spring 本地缓存】
Spring provided cacheable annotation since 3.1. It's very super convinient to use and can obviously boost application performance.
从3.1版本开始,Spring提供了cacheable注解。它使用起来非常方便,还可以很明显的提升应用性能。具体的,怎么使用呢?
First, create a cache bean.
首先,创建缓存bean。这里,我们设置一个三秒的本地缓存(写后3秒过期)。
- @Beanpublic Cache ephemeralCache() { return new ConcurrentMapCache(EPHEMERAL_CACHE, CacheBuilder.newBuilder() .expireAfterWrite(3, TimeUnit.SECONDS) .build().asMap(), false);}
Second, add @Cacheable to the existed method.
接下来,给想要使用缓存的已有方法加上@Cacheable注解
- @Cacheable(cacheNames = AppCacheConfig.EPHEMERAL_CACHE, key = "{#root.methodName, 'test'}")public Integer genId() { int i = ai.incrementAndGet(); System.out.println(String.format("populate cache %s", LocalDateTime.now())); return i;}
Finally, enjoy!
然后,尽情体验缓存带来的快乐!
- ExecutorService exe = Executors.newWorkStealingPool();exe.submit(() -> { while (true) { cacheApi.genId(); Thread.sleep(50); }});
See output of below codes
下面的日志和最初的缓存设定完全吻合
- 2017-06-08 10:17:49.990 INFO 12460 --- [ main] com.loops.nbs.Application : Started Application in 7.42 seconds (JVM running for 7.973)
- populate cache 2017-06-08T10:17:52.372
- populate cache 2017-06-08T10:17:55.379
- populate cache 2017-06-08T10:17:58.387
- populate cache 2017-06-08T10:18:01.394
- populate cache 2017-06-08T10:18:04.402
- populate cache 2017-06-08T10:18:07.409
- populate cache 2017-06-08T10:18:10.417
- populate cache 2017-06-08T10:18:13.426
2. Guava cache 【Guava 本地缓存】
Guava is a set of libraries provided by Google. In memory cache is part of it.
Guava是Google提供的一套工具库。这里只讨论其中的缓存部分。
- public class GuavaCacheExample {
- static Cache<Integer, Integer> cache = CacheBuilder.newBuilder()
- .expireAfterWrite(2, TimeUnit.SECONDS)
- .recordStats()
- .build();
- static LoadingCache<Integer, Integer> loadingCache = CacheBuilder.newBuilder()
- .expireAfterWrite(2, TimeUnit.SECONDS)
- .build(new CacheLoader<Integer, Integer>() {
- @Override
- public Integer load(Integer key) throws Exception {
- System.out.println("populate cache");
- return key * 10;
- }
- });
- static ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);
- public static void main(String[] args) throws Exception {
- useNormalCache();
- }
- static void useNormalCache() throws Exception {
- scheduler.scheduleWithFixedDelay(() -> System.out.println(cache.stats()), 0, 1, TimeUnit.SECONDS);
- for (int i = 0; i < 10; i++) {
- System.out.println(cache.get(0, () -> {
- Thread.sleep(500);
- return 10;
- }));
- Thread.sleep(300);
- }
- }
- static void useLoadingCache() throws Exception {
- for (int i = 0; i < 10; i++) {
- System.out.println(loadingCache.get(1));
- Thread.sleep(300);
- }
- }
- }
Usually, there are 2 types of caches
1) Cache
Cache is basic usage, it provides method to get and put cache.
Cache是基础用法,提供了get和put的操作。
2) LoadingCache
Loading cache requires a CacheLoader when cache is created. Every time when cache is expired or nonexisted, the load method will be called automatically and generate cache.
So user doesn't have to manually put cache back after cache is expired.
Loading cache在创建的时候需要给定 CacheLoader。如果缓存过期或者不存在,CacheLoader 的 load 方法就会被调用到并产生缓存。这样,用户就不用去关心那些过期的缓存项了。
Besides, we can add recordStats() when creating a cache object. Later we can monitor the cache usage by calling cache.stats()
Cache is created twice during the test, correspondingly, totalLoadTime (mesured in nano seconds) changed twice.
除此,我们还可以在创建缓存时使用 recordStats 来记录缓存的使用情况。之后用 cache 的 stats() 方法来观察。
测试过程中,我们的缓存产生了2次,totalLoadTime (纳秒为单位)也很好的佐证了这一点。
- CacheStats{hitCount=0, missCount=0, loadSuccessCount=0, loadExceptionCount=0, totalLoadTime=0, evictionCount=0}
- 10
- 10
- CacheStats{hitCount=1, missCount=1, loadSuccessCount=1, loadExceptionCount=0, totalLoadTime=502324777, evictionCount=0}
- 10
- 10
- 10
- CacheStats{hitCount=4, missCount=1, loadSuccessCount=1, loadExceptionCount=0, totalLoadTime=502324777, evictionCount=0}
- 10
- 10
- CacheStats{hitCount=6, missCount=1, loadSuccessCount=1, loadExceptionCount=0, totalLoadTime=502324777, evictionCount=1}
- 10
- 10
- 10
- CacheStats{hitCount=8, missCount=2, loadSuccessCount=2, loadExceptionCount=0, totalLoadTime=1002802169, evictionCount=1}
3. Caffeine cache 【Caffeine 本地缓存】
Caffeine cache is an optimized cache for Java 8. It has similar apis to guava but provide higher performance, which makes it easy to migrate from Gauva to Caffeine.
See this report for more details https://github.com/ben-manes/caffeine/wiki/Benchmarks
咖啡因缓存是专为Java 8而生的。使用上和Guava很像(很容易迁移),但它在多线程情况下性能更高。上面有一个跑分链接,比较了各种本地缓存的性能。
- public class CaffeineCacheExample {
- static Cache<Integer, Integer> cache = Caffeine.newBuilder()
- .expireAfterWrite(2, TimeUnit.SECONDS)
- .recordStats()
- .build();
- static LoadingCache<Integer, Integer> loadingCache = Caffeine.newBuilder()
- .expireAfterWrite(2, TimeUnit.SECONDS)
- .build(new CacheLoader<Integer, Integer>() {
- @Override
- public Integer load(Integer key) throws Exception {
- System.out.println("populate cache");
- return key * 10;
- }
- });
- static ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);
- public static void main(String[] args) throws Exception {
- useNormalCache();
- }
- static void useNormalCache() throws Exception {
- scheduler.scheduleWithFixedDelay(() -> System.out.println(cache.stats()), 0, 1, TimeUnit.SECONDS);
- for (int i = 0; i < 10; i++) {
- System.out.println(cache.get(0, k -> {
- try {
- Thread.sleep(500);
- } catch (InterruptedException ex) {
- //ignore
- }
- return 10;
- }));
- Thread.sleep(300);
- }
- }
- static void useLoadingCache() throws Exception {
- for (int i = 0; i < 10; i++) {
- System.out.println(loadingCache.get(1));
- Thread.sleep(300);
- }
- }
- }
There exists slight difference between Guava and Caffeine
使用上,有一点小小的区别
1) Builder: Guava use CacheBuilder to create new builder while Caffeine use Caffeine.
创建Builder时,Guava使用CacheBuilder,咖啡因使用Caffeine
2) Get method: Guava pass callable for get method
两者的Get方法第二个参数略有区别Guava 传入Callable, 内部可以抛出异常,无需写冗余的try catch,而咖啡因传入Function,必须手动捕获可能的异常
- V get(K var1, Callable<? extends V> var2) throws ExecutionException;
Caffeine pass Function for get method
- @CheckForNull
- V get(@Nonnull K var1, @Nonnull Function<? super K, ? extends V> var2);
Callable itself throws exception so we don't have to try catch the exception in the block while Function tolerates no exception.
- @FunctionalInterface
- public interface Callable<V> {
- /**
- * Computes a result, or throws an exception if unable to do so.
- *
- * @return computed result
- * @throws Exception if unable to compute a result
- */
- V call() throws Exception;
- }
- @FunctionalInterface
- public interface Function<T, R> {
- /**
- * Applies this function to the given argument.
- *
- * @param t the function argument
- * @return the function result
- */
- R apply(T t);
A comparison of local caches (1) 【本地缓存之比较 (1)】的更多相关文章
- A comparison of local caches (2) 【本地缓存之比较 (2)】
接上一篇: A comparison of local caches (1) [本地缓存之比较 (1)] This article will compare the asynchronous loca ...
- 八、React实战:可交互待办事务表(表单使用、数据的本地缓存local srtorage、生命同期函数(页面加载就会执行函数名固定为componentDidMount()))
一.项目功能概述 示例网址:http://www.todolist.cn/ 功能: 输入待做事项,回车,把任务添加到 [正在进行] [正在进行] 任务,勾选之后,变成已[经完成事项] [已完成事务], ...
- spring boot: 用redis的消息订阅功能更新应用内的caffeine本地缓存(spring boot 2.3.2)
一,为什么要更新caffeine缓存? 1,caffeine缓存的优点和缺点 生产环境中,caffeine缓存是我们在应用中使用的本地缓存, 它的优势在于存在于应用内,访问速度最快,通常都不到1ms就 ...
- spring boot:使用spring cache+caffeine做进程内缓存(本地缓存)(spring boot 2.3.1)
一,为什么要使用caffeine做本地缓存? 1,spring boot默认集成的进程内缓存在1.x时代是guava cache 在2.x时代更新成了caffeine, 功能上差别不大,但后者在性能上 ...
- Java8简单的本地缓存实现
原文出处:lukaseder Java8简单的本地缓存实现 这里我将会给大家演示用ConcurrentHashMap类和lambda表达式实现一个本地缓存.因为Map有一个新的方法,在 ...
- iOS五种本地缓存数据方式
iOS五种本地缓存数据方式 iOS本地缓存数据方式有五种:前言 1.直接写文件方式:可以存储的对象有NSString.NSArray.NSDictionary.NSData.NSNumber,数据 ...
- ImageLoader(多线程网络图片加载)+本地缓存 for windowsphone 7
搞了好长一阵子wp,做点好事. C/S手机app中应用最多的是 获取网络图片,缓存到本地,展示图片 本次主要对其中的delay:LowProfileImageLoader进行修改,在获取图片的时候, ...
- lua模块demo(redis,http,mysql,cjson,本地缓存)
1. lua模块demo(redis,http,mysql,cjson,本地缓存) 1.1. 配置 在nginx.conf中设置lua_shared_dict my_cache 128m; 开启ngi ...
- ASP.NET MVC深入浅出(被替换) 第一节: 结合EF的本地缓存属性来介绍【EF增删改操作】的几种形式 第三节: EF调用普通SQL语句的两类封装(ExecuteSqlCommand和SqlQuery ) 第四节: EF调用存储过程的通用写法和DBFirst模式子类调用的特有写法 第六节: EF高级属性(二) 之延迟加载、立即加载、显示加载(含导航属性) 第十节: EF的三种追踪
ASP.NET MVC深入浅出(被替换) 一. 谈情怀-ASP.NET体系 从事.Net开发以来,最先接触的Web开发框架是Asp.Net WebForm,该框架高度封装,为了隐藏Http的无状态 ...
随机推荐
- 【转】JDBC学习笔记(6)——获取自动生成的主键值&处理Blob&数据库事务处理
转自:http://www.cnblogs.com/ysw-go/ 获取数据库自动生成的主键 我们这里只是为了了解具体的实现步骤:我们在插入数据的时候,经常会需要获取我们插入的这一行数据对应的主键值. ...
- jquery使用CSS3实现文字动画效果插件Textillate.js
Textillate是一款基于jquery的使用CSS3实现文字动画的小巧插件.Textillate.js集成了一些很棒的使用CSS3动画效果的 JavaScript 库,您可非常轻轻松地把这些动画效 ...
- JavaWeb的国际化(17/4/8)
国际化的缺点: 因为文字不同,所以带来的排版问题一样严重,通常都是重新在写一个网站反而更加清晰,快捷 1:需要从浏览器中获取到浏览器语言(Accept-Language) 2:利用locale获取 ...
- C/C++对bool operator < (const p &a)const的认识,运算符重载详解(杂谈)
下面来进行这段代码的分析: struct node { //定义一个结构体node(节点) int x; int y; int len; //node中有3个成员变量x,y,l ...
- uoj#179 线性规划
这是一道模板题. 本题中你需要求解一个标准型线性规划: 有nn个实数变量x1,x2,⋯,xnx1,x2,⋯,xn和mm条约束,其中第ii条约束形如∑nj=1aijxj≤bi∑j=1naijxj≤bi. ...
- 【算法系列学习】SPFA邻接表最短路 [kuangbin带你飞]专题四 最短路练习 F - Wormholes
https://vjudge.net/contest/66569#problem/F 题意:判断图中是否存在负权回路 首先,介绍图的邻接表存储方式 数据结构:图的存储结构之邻接表 邻接表建图,类似于头 ...
- 蓝桥杯-马虎的算式-java
/* (程序头部注释开始) * 程序的版权和版本声明部分 * Copyright (c) 2016, 广州科技贸易职业学院信息工程系学生 * All rights reserved. * 文件名称: ...
- 全易通人事考勤工资验厂管理系统软件创建连接SQL2000数据库的操作方法和说明
全易通人事考勤工资验厂管理系统软件创建连接SQL2000数据库的操作方法和说明.全易通人事考勤工资验厂管理系统软件,有2种数据库,一个是ACCESS,另一个是SQL.不过由于ACCESS数据库比较小, ...
- luogu P1007 独木桥
序:难度标签是普及-,便觉得应该非常简单,结果发现有一个弯半天没绕过来,所以认为这道题对于第一次做的人来讲还是很是比较有意义的. 题目描述: 长度为len的桥上有n个士兵,你不知道他们的初始方向.已知 ...
- NodeJS+Express+MongoDB 简单实现数据录入及回显展示【适合新人刚接触学习】
近期在看NodeJS相关 不得不说NodeJS+Express 进行网站开发是很不错,对于喜欢玩JS的来说真是很好的一种Web开发组合 在接触NodeJS时受平时Java或者C#中API接口等开发的思 ...