高性能 Java 本地缓存 Caffeine 框架介绍及在 SpringBoot 中的使用

在现代应用程序中,缓存是一种重要的性能优化技术,它可以显著减少数据访问延迟,降低服务器负载,提高系统的响应速度。特别是在高并发的场景下,合理地使用缓存能够有效提升系统的稳定性和效率。

Caffeine 是一个高性能的 Java 本地缓存库,以其近乎最佳的命中率和灵活的配置选项在众多缓存解决方案中脱颖而出。它的设计目标是提供一个简单但功能强大的缓存实现,支持多种缓存策略,包括基于大小的淘汰、基于时间的过期、弱引用和软引用等。


文章目录

      • 1、Caffeine Cache 介绍
      • 2、Caffeine Cache 使用示例
        • 2.1、依赖引入
        • 2.2、示例代码
      • 3、Caffeine Cache 在 SpringBoot 中的使用
        • 3.1、依赖引入
        • 3.2、配置文件
        • 3.3、启用缓存
        • 3.4、配置缓存管理器
        • 3.5、使用缓存


1、Caffeine Cache 介绍

Caffeine 是一个用于 Java 的缓存库,其设计目标是高性能,和近乎最佳的命中率。缓存库的主要功能是存储和快速检索数据,以减少直接访问数据源(例如数据库或远程服务)的次数,从而提高应用程序的性能。Caffeine 的 “高性能” 表示它在处理缓存操作时速度很快,并且占用的资源较少。“近乎最佳的命中率” 指的是 Caffeine 在缓存命中(即请求的数据已经存在于缓存中)方面表现非常出色。高命中率意味着大多数数据请求都可以直接从缓存中获得,而无需访问原始数据源,从而大大提高了效率和响应速度。

image-20240726183556025

Caffeine 是受 Google Guava 提供内存缓存 API 的启发。借着 Guava 的思想优化了算法发展而来。

Caffeine 缓存与并发映射(ConcurrentMap)类似,但不完全相同。最根本的区别在于,并发映射会保存所有添加到其中的元素,直到它们被显式地移除。而缓存通常配置为自动驱逐条目,以限制其内存占用。在某些情况下,即使缓存不驱逐条目,LoadingCache 或 AsyncLoadingCache 也可能非常有用,因为它们可以自动加载缓存。

Caffeine 提供灵活的构建方式来创建具有以下可选功能组合的缓存:

  • 自动加载缓存条目,可以选择异步方式
  • 基于大小的驱逐,当超过最大值时根据使用频率和最近使用时间进行驱逐
  • 基于时间的条目过期,依据最后访问时间或最后写入时间进行测量
  • 当首次请求陈旧条目时异步刷新
  • 键自动包装成弱引用
  • 值自动包装成弱引用或软引用
  • 驱逐(或其他方式移除)条目的通知
  • 将写操作传播到外部资源
  • 累积缓存访问统计信息

为了改进集成,扩展模块中提供了 JSR-107 JCache 和 Guava 适配器。JSR-107 标准化了基于 Java 6 的 API,以最小化特定供应商代码的代价来实现功能和性能。Guava 的 Cache 是 Caffeine 的前身库,这些适配器提供了一个简单的迁移策略。


2、Caffeine Cache 使用示例

2.1、依赖引入

Caffeine 可以通过 Maven 或 Gradle 依赖引入,对于 Java 11 或更高版本,使用 3.x 其他版本则使用 2.x

Maven 依赖:

<!-- https://mvnrepository.com/artifact/com.github.ben-manes.caffeine/caffeine -->
<dependency><groupId>com.github.ben-manes.caffeine</groupId><artifactId>caffeine</artifactId><version>2.9.3</version>
</dependency>
2.2、示例代码

可以参照以下示例代码来使用 CaffeineCache:

import com.github.benmanes.caffeine.cache.Cache;
import com.github.benmanes.caffeine.cache.Caffeine;import java.util.concurrent.TimeUnit;public class CaffeineCacheExample {public static void main(String[] args) {// 初始化缓存Cache<String, String> cache = Caffeine.newBuilder()// 设置最大缓存条目数.maximumSize(100)// 设置条目在最后一次访问后10分钟过期.expireAfterAccess(10, TimeUnit.MINUTES)// 设置条目在写入后10分钟过期.expireAfterWrite(10, TimeUnit.MINUTES)// 设置键使用弱引用.weakKeys()// 设置值使用软引用.softValues()// 构建缓存.build();// 添加一些条目到缓存中cache.put("key1", "value1");cache.put("key2", "value2");// 获取并打印缓存条目System.out.println("key1: " + cache.getIfPresent("key1"));System.out.println("key2: " + cache.getIfPresent("key2"));// 模拟访问 "key1"cache.getIfPresent("key1");// 等待一段时间后再获取缓存条目try {// 等待6秒Thread.sleep(6000); } catch (InterruptedException e) {Thread.currentThread().interrupt();}// 获取并打印缓存条目System.out.println("key1 (after 6 seconds): " + cache.getIfPresent("key1"));System.out.println("key2 (after 6 seconds): " + cache.getIfPresent("key2"));// 模拟再等待时间超过10分钟,检查过期try {// 等待10分钟Thread.sleep(600000); } catch (InterruptedException e) {Thread.currentThread().interrupt();}// 获取并打印缓存条目,预期缓存条目已过期System.out.println("key1 (after 10 minutes): " + cache.getIfPresent("key1"));System.out.println("key2 (after 10 minutes): " + cache.getIfPresent("key2"));}
}

3、Caffeine Cache 在 SpringBoot 中的使用

在 Spring Boot 应用中,可以通过配置文件和注解来简化对 Caffeine 缓存的集成和管理。

3.1、依赖引入

pom.xml 文件中添加 Caffeine 和 Spring Boot 缓存依赖:

<dependencies><!-- Spring Boot Starter Cache --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-cache</artifactId></dependency><!-- Caffeine Cache --><dependency><groupId>com.github.ben-manes.caffeine</groupId><artifactId>caffeine</artifactId></dependency>
</dependencies>
3.2、配置文件

application.yml 中配置 Caffeine 缓存属性:

spring:cache:type: caffeinecache-names: myCachecaffeine:spec: maximumSize=100,expireAfterWrite=10m,expireAfterAccess=10m

这里 spec 属性用于设置缓存策略,包括最大缓存大小 (maximumSize)、写入后的过期时间 (expireAfterWrite),以及访问后的过期时间 (expireAfterAccess)。

3.3、启用缓存

在 Spring Boot 应用主类上添加 @EnableCaching 注解:

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cache.annotation.EnableCaching;@SpringBootApplication
@EnableCaching
public class ExampleApplication {public static void main(String[] args) {SpringApplication.run(ExampleApplication.class, args);}
}
3.4、配置缓存管理器

创建一个配置类来自定义缓存管理器:

import com.github.benmanes.caffeine.cache.Caffeine;
import org.springframework.cache.annotation.EnableCaching;
import org.springframework.cache.caffeine.CaffeineCacheManager;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;import java.util.concurrent.TimeUnit;@Configuration
@EnableCaching
public class CacheConfig {@Beanpublic CaffeineCacheManager cacheManager() {CaffeineCacheManager cacheManager = new CaffeineCacheManager("myCache");cacheManager.setCaffeine(Caffeine.newBuilder().maximumSize(100).expireAfterWrite(10, TimeUnit.MINUTES).expireAfterAccess(10, TimeUnit.MINUTES));return cacheManager;}
}
3.5、使用缓存

在需要缓存的方法上添加 @Cacheable 注解:

import org.springframework.cache.annotation.Cacheable;
import org.springframework.stereotype.Service;@Service
public class MyService {@Cacheable("myCache")public String getData(String key) {// 模拟获取数据的过程return "Data for " + key;}
}
  • @Cacheable("myCache"):注解告诉 Spring 该方法的返回值应该被缓存。缓存的名字是 myCache
  • @RequestParam String key:注解从请求参数中获取 key 的值并传递给 getData 方法;
  • 缓存命中:如果缓存中存在对应 key 的值,则直接返回缓存值,否则执行方法体并将结果存入缓存。

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

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

宝塔单ip,新建多站点

报错如上&#xff1a; 那么如何新建多站点呢 先随便写个名字上去&#xff0c;然后再重新绑定别的端口… 这个时候访问99端口即可 。 如果是有域名&#xff0c;则不需要这样做 、直接80端口也可以多站点

OSPF配置与分析

前言&#xff1a;本博客仅作记录学习使用&#xff0c;部分图片出自网络&#xff0c;如有侵犯您的权益&#xff0c;请联系删除 一、相关知识 开放最短路径优先&#xff08;Open Shortest Path First&#xff0c;OSPF&#xff09;基于链路状态算法&#xff0c;是一个内部网关协议…

gitee设置ssh公钥密码避免频繁密码验证

gitee中可以创建私有项目&#xff0c;但是在clone或者push都需要输入密码&#xff0c; 比较繁琐。 公钥则可以解决该问题&#xff0c;将私钥放在本地&#xff0c;公钥放在gitee上&#xff0c;当对项目进行操作时带有的私钥会在gitee和公钥进行验证&#xff0c;避免了手动输入密…

三目操作符

双目操作符有、-、*、/、% 单目操作符有--、、、- 三目操作符有表达式1&#xff1f;表达式2&#xff1a;表达式3 如果表达式1为真则表达式2计算否则表达式3计算&#xff0c;计算结果为整个表达式的结果 #include<stdio.h> int main() {int a 0;int b 0;scanf(&quo…

Web开发:使用数据库工具Navicat技巧合集

1.EXCEL批量导入数据 打开Navicat准备导入数据&#xff0c;点击导入 选择excel 字段名行应该写0&#xff08;下图错误&#xff09; 下一步&#xff0c;表已经用语法建好了&#xff0c;因此不用打勾 配置好字段&#xff0c;下一步&#xff0c;点击开始即可。 2.数据表从一个服…

R包:plot1cell单细胞可视化包

介绍 plot1cell是用于单细胞数据seurat数据对象的可视化包。 安装 ## You might need to install the dependencies below if they are not available in your R library. bioc.packages <- c("biomaRt","GenomeInfoDb","EnsDb.Hsapiens.v86&qu…

【React 】开发环境搭建详细指南

文章目录 一、准备工作1. 安装 Node.js 和 npm2. 选择代码编辑器 二、创建 React 项目1. 使用 Create React App2. 手动配置 React 项目 三、集成开发工具1. ESLint 和 Prettier2. 使用 Git 进行版本控制 在现代前端开发中&#xff0c;React 是一个非常流行的框架&#xff0c;用…

与众不同的社交体验:Facebook的新功能与新变化

在快速变化的社交媒体领域&#xff0c;Facebook不断引入创新功能和变化&#xff0c;以满足用户日益增长的需求&#xff0c;并提供与众不同的社交体验。从增强现实到数据隐私&#xff0c;Facebook的新功能和更新正在塑造一个全新的社交平台。本文将深入探讨这些新功能和变化&…

昇思25天学习打卡营第33天|共赴算力时代

文章目录 一、平台简介二、深度学习模型2.1 处理数据集2.2 模型训练2.3 加载模型 三、共赴算力时代 一、平台简介 昇思大模型平台&#xff0c;就像是AI学习者和开发者的超级基地&#xff0c;这里不仅提供丰富的项目、模型和大模型体验&#xff0c;还有一大堆经典数据集任你挑。…

字节跳动推出端到端同声传译智能体;OpenAI 搜索引擎 SearchGPT 登场丨 RTE 开发者日报

开发者朋友们大家好&#xff1a; 这里是 「RTE 开发者日报」 &#xff0c;每天和大家一起看新闻、聊八卦。我们的社区编辑团队会整理分享 RTE&#xff08;Real-Time Engagement&#xff09; 领域内「有话题的 新闻 」、「有态度的 观点 」、「有意思的 数据 」、「有思考的 文…

2024算法、高性能计算与人工智能国际学术会议(AHPCAI 2024)

2024算法、高性能计算与人工智能国际学术会议&#xff08;AHPCAI 2024&#xff09; 2024 International Conference on Algorithms, High Performance Computing and Artificial Intelligence 2024年8月14-16日 | 中国-郑州 2024中国算力大会正在发起“算力中国最佳学术论文…

企业组织机构代码查询入口直达词令是什么?

企业组织机构代码查询入口直达词令是什么&#xff1f; 1、手机安装&#xff0c;打开词令关键词口令直达工具&#xff1b; 2、输入词令「组织机构代码查询」关键词直达口令&#xff1b; 3、搜索直达词令关联的目标企业组织机构代码查询入口&#xff1b; 4、输入营业执照上的18位…

基于DMASM镜像的DMDSC共享存储集群部署

DMv8镜像模式共享存储集群部署 环境说明 操作系统&#xff1a;centos7.6 服务器&#xff1a;2台虚拟机 达梦数据库版本&#xff1a;达梦V8 安装前准备工作 参考文档《DM8共享存储集群》-第11、12章节 参考文档《DM8_Linux服务脚本使用手册》 1、系统环境(all nodes) 1…

黄景仁,笔墨间的一抹清寒

黄景仁&#xff0c;字汉镛&#xff0c;一字仲则&#xff0c;号鹿菲子&#xff0c;生于乾隆十四年&#xff08;公元1749年&#xff09;&#xff0c;卒于乾隆四十八年&#xff08;公元1783年&#xff09;&#xff0c;享年仅35岁。他是宋代大诗人黄庭坚的后裔&#xff0c;出生于常…

【C++】初识C++基础篇·一(命名空间,函数重载,缺省参数,引用);

文章目录 前言1.输入与输出输出输入cin和scanf的对比 2.命名空间2.1namespace存在的意义2.2namespace的使用3.缺省参数4.函数重载重载函数的调用规则 5.引用 前言 我们先通过一段简单的代码来拉开C的序幕&#xff1b; //text.cpp #include<iostream> #include<stdio…

链路聚合加单臂路由

一、实验目的及拓扑 实验目的&#xff1a;在路由器及交换机之间建立链接聚合&#xff0c;交换机接入两台主机并通过路由器子接口自动分配IP地址&#xff0c;通过单臂路由实现两台主机互联 二、基本配置 1、交换机配置 [S1]vlan batch 10 20 [S1-Eth-Trunk1]dis th # interf…

matlab simulink 混合式VTOL-UAV动力学模型

1、内容简介 略 可以交流、咨询、答疑 2、内容说明 略混合式VTOL UAV是简单的将四旋翼和固定翼无人机结合在一起,不仅具备四旋翼飞行器的垂直起降,可悬停的特点,还具备固定翼飞行器的航程远,平飞速度快等特点,具有很重要的应用价值,在该飞行器布局中,四旋翼负责提供升力,工作…

《昇思25天学习打卡营第6天|ResNet50图像分类》

写在前面 从本次开始&#xff0c;接触一些上层应用。 本次通过经典的模型&#xff0c;开始本次任务。这里开始学习resnet50网络模型&#xff0c;应该也会有resnet18&#xff0c;估计18的模型速度会更快一些。 resnet 通过对论文的结论进行展示&#xff0c;说明了模型的功能&…

Ubuntu 22.04如何设置中文输入法

前言 近期整理了一下之前在ubuntu 22.04 中如何设置中文输入法的过程&#xff0c;对于本人比较适应读中文写中文来说&#xff0c;这是我安装后的第一步。 一、流程 1.1 安装中文语言包&#xff08;如果还未安装&#xff09; 首先是安装中文语言包&#xff0c;直接在终端输入…