Java缓存caffeine使用心得

文章目录

  • 添加依赖
  • 一、手动加载
    • 1,定义缓存
    • 2,写入缓存(增、改)
    • 3,获取大小
    • 4,模拟耗时操作
    • 5,移除缓存元素
    • 6,查询缓存(查)
    • 7,统计缓存
  • 二、同步加载
    • 1,定义缓存
    • 2,查询缓存
    • 3,刷新缓存
  • 三、异步加载
    • 1,定义缓存
    • 2,查找缓存
  • 参考博文

Caffeine是基于Java 8的高性能,接近最佳的缓存库,是Spring 5默认支持的Cache,底层数据存储采用ConcurrentHashMap,缓存和 Map 之间的一个根本区别在于缓存可以回收存储的 item。
在这里插入图片描述

添加依赖

<dependency>
  <groupId>com.github.ben-manes.caffeine</groupId>
  <artifactId>caffeine</artifactId>
  <version>2.9.3</version>
</dependency>

一、手动加载

1,定义缓存

Cache<String,String> cache = Caffeine.newBuilder()
// 初始容量
.initialCapacity(5)
// 写入间隔多久过期
.expireAfterWrite(10, TimeUnit.MINUTES)
// 最大缓存数,数字新写法(带下划线,增加可读性)
.maximumSize(10_000)
// 驱逐监听器
.evictionListener((String key,String str,RemovalCause cause) ->
                  System.out.printf("Key %s was evicted (%s)%n", key, cause))
// 移除监听器
.removalListener((String key,String str,RemovalCause cause) ->{
    System.out.printf("Key %s was removed (%s)%n", key, cause);
})
.build();

2,写入缓存(增、改)

cache.put("aaa","aac");
cache.put("bbb","bbd");

3,获取大小

此时如果我们把maximumSize调整为1,就是说最大缓存数就是1,而上一步我们添加2个缓存,这时调用estimatedSize方法获取到的长度还是2。因为虽然在执行缓存清理机制,不过该机制是异步的,所以需要加cleanUp方法后,再获取大小就是实际的大小

cache.cleanUp(); // 如果不加这行,打印的是size=2。加了这行打印size=1
long size = cache.estimatedSize();
System.out.println("size = " + size);
// size = 1

4,模拟耗时操作

验证过期策略

try {
    Thread.sleep(2000);
    // 这个sleep方法和Thread.sleep一样
    // TimeUnit.SECONDS.sleep(2000);
} catch (InterruptedException e) {
    throw new RuntimeException(e);
}

5,移除缓存元素

方法有三个(删)

// 移除某个元素
cache.invalidate("aaa");
// 移除所有缓存元素
cache.invalidateAll();
// 移除一组元素
cache.invalidateAll(keys);

6,查询缓存(查)

优先使用get方法,因为getIfPresent执行时非原子性,如果是高并发场景下可能会造成数据错乱

// 方法一:没有查找到的时候返回null
String ddd = cache.getIfPresent("ddd");
System.out.println("ddd="+ ddd);
// ddd=null
// 方法二:如果缓存不存在则生成缓存元素 并且写入缓存, 如果无法生成则返回null
String ccc = cache.get("ccc", k->"Hello");
System.out.println("ccc=" +ccc);
// ccc=Hello

7,统计缓存

long hitCount = cache.stats().hitCount();
long missCount = cache.stats().missCount();
System.out.println("统计缓存=>"+hitCount + "missCount:" +  missCount);
// 统计缓存=>hitCount=0;missCount:0

二、同步加载

1,定义缓存

LoadingCache<String,String> cache = Caffeine.newBuilder()
.maximumSize(10_000)
.expireAfterWrite(10,TimeUnit.MINUTES)
.build(key -> createData(key));

其中build参数也可以简写为

.build(TestCaffeineCache::createData);

createData函数如下,相当于给value一个默认值

private static String createData(String key){
return key + "value";
}

2,查询缓存

可以直接使用get(key)方法

String value = cache.get("hello");
System.out.println("key=="+value);
// key==hellovalue

如果在get之前先put了一下,那么打印出来的就是put的value值

cache.put("hello","world");
// key==world

批量查找缓存

Map<String,String> values  = cache.getAll(Arrays.asList("key1","key2"));
// 或者
Map<String,String> values  = cache.getAll(Stream.of("key1","key2").collect(Collectors.toList()));

3,刷新缓存

cache.refresh("key");

三、异步加载

1,定义缓存

AsyncLoadingCache<String ,String> cache = Caffeine.newBuilder()
.maximumSize(10_000)
.expireAfterWrite(10,TimeUnit.MINUTES)
.buildAsync( key -> createExpensiveStr(key));
private static String createExpensiveStr(String key){
return key + "expensiveValue";
}

2,查找缓存

可以发现除了返回值类型不太像,其他都和上一个同步加载差不多

CompletableFuture<String> hello = cache.get("hello");
hello.thenAccept( str ->{
    System.out.println("hello=="+str);
});
// hello==helloexpensiveValue
// 批量查找
CompletableFuture<Map<String,String>> values = cache.getAll(Arrays.asList("one","two"));
values.thenAccept( str ->{
            System.out.println("values=="+str);
        });
// values=={one=oneexpensiveValue, two=twoexpensiveValue}

参考博文

1,浅谈本地缓存的几种方案选型
2,Java内存缓存神器:Caffeine(咖啡因)
3,Caffeine缓存的简单介绍

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/615198.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

redis的双写一致性

双写一致性问题 1.先删除缓存或者先修改数据库都可能出现脏数据。 2.删除两次缓存&#xff0c;可以在一定程度上降低脏数据的出现。 3.延时是因为数据库一般采用主从分离&#xff0c;读写分离。延迟一会是让主节点把数据同步到从节点。 1.读写锁保证数据的强一致性 因为一般放…

TikTok海外运营:云手机的四种快速变现方法

随着TikTok用户基数的持续扩大&#xff0c;这个平台已成为全球创业者和品牌的新战场。其用户群接近20亿&#xff0c;并以年轻用户为主力军&#xff0c;市场渗透率逐年攀升。无论是大型组织、知名品牌&#xff0c;还是个人创业者&#xff0c;都无法忽视TikTok所带来的巨大商机。…

掌握SEO优化的关键:提升网站排名的秘籍(如何提高网站seo排名)

你是否曾经在搜索引擎上搜索过一个关键词&#xff0c;然后点击了排在前几位的网站&#xff1f;如果是&#xff0c;那么你已经体会到了SEO&#xff08;搜索引擎优化&#xff09;的威力。SEO是一项关键的网络营销策略&#xff0c;它能够让你的网站在搜索引擎中获得更高的排名&…

【Go】Go Swagger 生成和转 openapi 3.0.3

本文档主要描述在 gin 框架下用 gin-swagger 生成 swagger.json 的内容&#xff0c;中间猜的坑。以及&#xff0c;如何把 swagger 2.0 转成 openapi 3.0.3 下面操作均在项目根目录下执行 生成 swagger 2.0 import swagger go get -u github.com/swaggo/gin-swagger go get …

Android解放双手的利器之ViewBinding

文章目录 1. 背景2. ViewBinding是什么3. 开启ViewBinding功能4. 生成绑定类5. 使用ViewBinding5.1Activity 中使用5.2 Fragment 中使用5.3 ViewHolder 中使用 6. ViewBinding的优点7. 与 dataBinding 对比 1. 背景 写代码最繁琐的是什么&#xff1f;重复的机械操作。我们刚接…

Follow the Money:2023年最赚钱的十家国内芯片设计上市公司及其整体表现

作者&#xff1a;北京华兴万邦管理咨询有限公司 商瑞 马华 摘要&#xff1a;尽管相较2022年有所下滑&#xff0c;但2023年最赚钱的十家国内芯片设计上市公司的净利润总额超过了159家A股和港股上市内地半导体企业利润总额的55%&#xff0c;但是其市值之和仅占159家上市半导体…

前端开发工程师——ajax

express框架 终端输入 npm init --yes npm i express 请求报文/响应报文 // 1.引入express const express require(express);// 2.创建应用对象 const app express();// 3.创建路由规则 // request:是对请求报文的封装 // response&#xff1a;是对响应报文的封装 app.get(…

Docker学习笔记(一)安装Docker、镜像操作、容器操作、数据卷操作

文章目录 1 Docker介绍1.1 Docker的优势1.1.1 应用部署的环境问题1.1.2 Docker解决依赖兼容问题1.1.3 Docker解决操作系统环境差异1.1.4 小结 1.2 Docker和虚拟机的区别1.3 Docker架构1.3.1 镜像和容器1.3.2 DockerHub1.3.3 Docker架构 1.4 安装Docker1.4.1 卸载旧版本Docker&a…

SpringBoot 具体是做什么的?

Spring Boot是一个用于构建独立的、生产级别的、基于Spring框架的应用程序的开源框架。它的目标是简化Spring应用程序的开发和部署过程&#xff0c;通过提供一种快速、便捷的方式来创建Spring应用程序&#xff0c;同时保持Spring的灵活性和强大特性。 1. 简化Spring应用程序开…

MVC:一种设计模式而非软件架构

在软件开发领域&#xff0c;MVC&#xff08;Model-View-Controller&#xff09;经常被提及&#xff0c;但很多人对其定位存在误解。本文将澄清一个常见的误区&#xff1a;MVC是一种设计模式&#xff0c;而非软件架构。 一、MVC简介 MVC&#xff0c;即模型&#xff08;Model&a…

导电材料——分类、性能与性质

本篇为西安交通大学本科课程《电气材料基础》的笔记。 导电材料指的是能在电场下传导电流的材料。导体价电子所在能带为半满带&#xff0c;且相邻能级间隔小&#xff0c;外电场下电子很容易从低能级跃迁到高能级上&#xff0c;大量的电子很容易获得能量进行共有化运动&#xf…

Linux日常管理和服务器配置(二)

一、在系统中配置FTP服务器&#xff1a; 准备工作&#xff1a; a.下载ftp命令 sudo apt install vsftpd 可以先用命令更新一下库 sudo apt-get update 接着输入 systemctl status vsftpd 检查ftp运行状态 然后进入vsftpd.conf文件中修改write为 vim /etc/vsftpdf.conf …

本安防爆手机在化工巡检作业中的作用

在充满潜在危险的化工环境中&#xff0c;巡检作业不仅需要高度的专业性和精确性&#xff0c;更要求每一位巡检人员能够在保障自身安全的前提下&#xff0c;高效地完成各项任务。在这一背景下&#xff0c;防爆手机以其独特的优势&#xff0c;成为了化工巡检作业中不可或缺的利器…

【Android】Kotlin学习之Lambda表达式

java和kotlin对比 Lambda语法 Lambda隐形参数 it 也可以不使用指定的名称it, 可以 自定义 Lambda 使用下划线

Simulink建模的基础知识(精)

01--Stateflow建模 1.背景 很多时候&#xff0c;我们在拿到需求之后搭建模型&#xff0c;到底是选用Simulink还是Stateflow&#xff0c;经常会不够清晰&#xff0c;也跟自己掌握的技能有关系&#xff0c;有些人接触Simulink较多&#xff0c;不管什么逻辑都要Simulink来做。其…

【前端】桌面版docker并部署前端项目

环境 win10专业版 2004 , 需科学 官网下载安装包并安装4.29.0版本 终端输入 wsl --installdocker桌面版和模拟器只能选一个&#xff0c;不然一直转圈圈 镜像配置加速&#xff0c;在settings—>docker engine下 {"builder": {"gc": {"defaultKee…

基于JSP动漫论坛的设计与实现(四)

目录 功能模块测试 6.1 测试概述及所用方案 6.1.1软件测试概述 6.1.3 测试的步骤 6.1.4 测试的主要内容 6.1.5 测试方案 6.1.6测试设计 6.2 前端功能测试 6.2.1 登录功能测试 6.2.2 注册功能测试 6.2.3 发帖功能测试 6.2.4 回复帖子测试 6.3 后台功能测试 6…

【每日刷题】Day37

【每日刷题】Day37 &#x1f955;个人主页&#xff1a;开敲&#x1f349; &#x1f525;所属专栏&#xff1a;每日刷题&#x1f34d; &#x1f33c;文章目录&#x1f33c; 1. 2391. 收集垃圾的最少总时间 - 力扣&#xff08;LeetCode&#xff09; 2. 1614. 括号的最大嵌套深度…

【吴恩达机器学习-week2】多个变量的特征缩放和学习率问题

特征缩放和学习率&#xff08;多变量&#xff09; 目标 利用上一个实验中开发的多变量例程在具有多个特征的数据集上运行梯度下降探索学习率对梯度下降的影响通过 Z 分数归一化进行特征缩放&#xff0c;提高梯度下降的性能 import numpy as np np.set_printoptions(precisio…

分布式链路追踪 Zipkin+Sleuth(8)

项目的源码地址 Spring Cloud Alibaba 工程搭建&#xff08;1&#xff09; Spring Cloud Alibaba 工程搭建连接数据库&#xff08;2&#xff09; Spring Cloud Alibaba 集成 nacos 以及整合 Ribbon 与 Feign 实现负载调用&#xff08;3&#xff09; Spring Cloud Alibaba Ribbo…