0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

基于java开发的缓存框架jetcache简介

马哥Linux运维 来源:稀土掘金技术社区 2023-09-07 10:36 次阅读

0. 引言

在实际应用中,并不是单一的使用本地缓存或者redis,更多是组合使用来满足不同的业务场景,于是如何优雅的组合本地缓存和远程缓存就成了我们要研究的问题,而这一点,阿里开源的jetcache组件帮我们实现了

1. jetcache简介

jetcache是阿里开源的基于java开发的缓存框架,支持多种缓存类型:本地缓存、分布式缓存、多级缓存。能够满足不同业务场景的缓存需求。

jetcache具有上手简单、性能高效、拓展性强的特点。支持缓存预热 、缓存key前缀等功能。结合spring-cache使用,可以实现十分优雅的缓存类型切换

2. jetcache使用

1、引入依赖,这里我们使用sringboot项目框架,同时使用redis作为远程缓存。于是我们引入jetcache-starter-redis依赖,这里我的springboot版本为2.6.13

如果是非springboot项目可以参考官网说明配置

8f7e9880-4cb5-11ee-a25d-92fbcf53809c.png


com.alicp.jetcache
jetcache-starter-redis
2.7.0


 

redis.clients
jedis
4.3.1

对应的版本说明如下:springboot与jetcache版本关系

8fa17d64-4cb5-11ee-a25d-92fbcf53809c.png

2、修改配置文件,配置redis地址和线程数

jetcache:
#统计间隔,0表示不统计,开启后定期在控制台输出缓存信息
statIntervalMinutes:15
#是否把cacheName作为远程缓存key前缀
areaInCacheName:false
#本地缓存配置
local:
default:#default表示全部生效,也可以指定某个cacheName
#本地缓存类型,其他可选:caffeine/linkedhashmap
type:linkedhashmap
keyConvertor:fastjson
#远程缓存配置
remote:
default:#default表示全部生效,也可以指定某个cacheName
type:redis
#key转换器方式n
keyConvertor:fastjson
broadcastChannel:projectA
#redis序列化方式
valueEncoder:java
valueDecoder:java
#redis线程池
poolConfig:
minIdle:5
maxIdle:20
maxTotal:50
#redis地址与端口
host:127.0.0.1
port:6379

更详细的参数配置

8fb5b57c-4cb5-11ee-a25d-92fbcf53809c.png

3、启动类添加注解@EnableCreateCacheAnnotation,开启缓存,添加@EnableMethodCache(basePackages = "com.example.jetcachedemo")注解,配置缓存方法扫描路径

4、使用缓存可以通过三种方式:

方式一(推荐)AOP模式:通过@Cached,@CacheUpdate,@CacheInvalidate注解

@RestController
@RequestMapping("user")
publicclassUserController{

@GetMapping("getRemote")
@Cached(name="userCache:",key="#id",expire=3600,timeUnit=TimeUnit.SECONDS,cacheType=CacheType.REMOTE)
publicUsergetRemote(Longid){
//直接新建用户,模拟从数据库获取数据
Useruser=newUser();
user.setId(id);
user.setName("用户remote"+id);
user.setAge(23);
user.setSex(1);
System.out.println("第一次获取数据,未走缓存:"+id);
returnuser;
}

@GetMapping("getLocal")
@Cached(name="userCache:",key="#id",expire=3600,timeUnit=TimeUnit.SECONDS,cacheType=CacheType.LOCAL)
publicUsergetLocal(Longid){
//直接新建用户,模拟从数据库获取数据
Useruser=newUser();
user.setId(id);
user.setName("用户local"+id);
user.setAge(23);
user.setSex(1);
System.out.println("第一次获取数据,未走缓存:"+id);
returnuser;
}

@GetMapping("getBoth")
@Cached(name="userCache:",key="#id",expire=3600,timeUnit=TimeUnit.SECONDS,cacheType=CacheType.BOTH)
publicUsergetBoth(Longid){
//直接新建用户,模拟从数据库获取数据
Useruser=newUser();
user.setId(id);
user.setName("用户both"+id);
user.setAge(23);
user.setSex(1);
System.out.println("第一次获取数据,未走缓存:"+id);
returnuser;
}

@PostMapping("updateUser")
@CacheUpdate(name="userCache:",key="#user.id",value="#user")
publicBooleanupdateUser(@RequestBodyUseruser){
//TODO更新数据库
returntrue;
}

@PostMapping("deleteUser")
@CacheInvalidate(name="userCache:",key="#id")
publicBooleandeleteUser(Longid){
//TODO从数据库删除
returntrue;
}

}

这里要注意实体类User一定要实现序列化,即声明Serializable

@Data
publicclassUserimplementsSerializable{

privateLongid;
privateStringname;
privateIntegerage;
privateIntegersex;
}

方式二 API模式:通过@CreateCache,注:在jetcache 2.7 版本CreateCache注解已废弃,不推荐使用

@RestController
@RequestMapping("user2")
publicclassUser2Controller{

@CreateCache(name="userCache:",expire=3600,timeUnit=TimeUnit.SECONDS,cacheType=CacheType.BOTH)
privateCacheuserCache;

@GetMapping("get")
publicUserget(Longid){
if(userCache.get(id)!=null){
return(User)userCache.get(id);
}
Useruser=newUser();
user.setId(id);
user.setName("用户both"+id);
user.setAge(23);
user.setSex(1);
userCache.put(id,user);
System.out.println("第一次获取数据,未走缓存:"+id);
returnuser;
}

@PostMapping("updateUser")
publicBooleanupdateUser(@RequestBodyUseruser){
//TODO更新数据库
userCache.put(user.getId(),user);
returntrue;
}

@PostMapping("deleteUser")
publicBooleandeleteUser(Longid){
//TODO从数据库删除
userCache.remove(id);
returntrue;
}

}

方式三 高级API模式:通过CacheManager,2.7 版本才可使用

(1)添加依赖


com.alibaba
fastjson
2.0.25

(2)书写配置类

@Configuration
publicclassJetcacheConfig{

@Autowired
privateCacheManagercacheManager;
privateCacheuserCache;

@PostConstruct
publicvoidinit(){
QuickConfigqc=QuickConfig.newBuilder("userCache:")
.expire(Duration.ofSeconds(3600))
.cacheType(CacheType.BOTH)
//本地缓存更新后,将在所有的节点中删除缓存,以保持强一致性
.syncLocal(false)
.build();
userCache=cacheManager.getOrCreateCache(qc);
}

@Bean
publicCachegetUserCache(){
returnuserCache;
}
}

(3)调用代码

@RestController
@RequestMapping("user3")
publicclassUser3Controller{

@Autowired
JetcacheConfigjetcacheConfig;
@Autowired
privateCacheuserCache;

@GetMapping("get")
publicUserget(Longid){
if(userCache.get(id)!=null){
return(User)userCache.get(id);
}
Useruser=newUser();
user.setId(id);
user.setName("用户both"+id);
user.setAge(23);
user.setSex(1);
userCache.put(id,user);
System.out.println("第一次获取数据,未走缓存:"+id);
returnuser;
}

@PostMapping("updateUser")
publicBooleanupdateUser(@RequestBodyUseruser){
//TODO更新数据库
userCache.put(user.getId(),user);
returntrue;
}

@PostMapping("deleteUser")
publicBooleandeleteUser(Longid){
//TODO从数据库删除
userCache.remove(id);
returntrue;
}

}

多级缓存的形式,会先从本地缓存获取数据,本地获取不到会从远程缓存获取

5、启动redis,启动演示项目

注意,如果启动出现NoClassDefFoundError: redis/clients/util/Pool或NoClassDefFoundError: redis/clients/jedis/UnifiedJedis报错,说明springboot与jetcache版本不一致,对应关系可参考上述第一步中的说明 同时如果使用的是jetcache2.7.x版本,因为该版本中有jedis包的依赖,需要额外添加如下依赖,或者将jetcache版本将至2.6.5以下


redis.clients
jedis
4.3.1

3. 测试

3.1 方式一测试

1、访问localhost:8088/user/getRemote?id=1

8ff5ed0e-4cb5-11ee-a25d-92fbcf53809c.png

因为配置的是远程缓存,在redis中也能看到对应的key

902b0dd6-4cb5-11ee-a25d-92fbcf53809c.png

2、访问localhost:8088/user/getLocal?id=1,这个方法是从本地缓存获取的,现在只有远程缓存上有数据,我们调用发现缓存数据还是拿到了,这说明当我们在配置文件中配置了本地缓存和远程缓存后,方式一中本地缓存和远程缓存会自动相互调用

比如本地缓存有这个key,redis中没有,通过远程缓存方式访问时,会先从redis获取,如果没有会自动获取本地缓存,但是数据还是存储在本地缓存,并不会同步到redis上,这样更加灵活的实现了多级缓存架构

904d52ce-4cb5-11ee-a25d-92fbcf53809c.png

3.2 方式二测试

1、再测试下CreateCache的形式:localhost:8088/user2/get?id=4

9080d630-4cb5-11ee-a25d-92fbcf53809c.png

正常获取了,并且redis中也有了对应的值

90bbb124-4cb5-11ee-a25d-92fbcf53809c.png

而当我们把缓存方式更改为LOCAL后,再访问localhost:8088/user2/get?id=5

@CreateCache(name="userCache:",expire=3600,timeUnit=TimeUnit.SECONDS,cacheType=CacheType.LOCAL)

会发现redis中就没有对应缓存了,只在本地缓存存在,说明我们指定本地缓存的形式成功了

90d745e2-4cb5-11ee-a25d-92fbcf53809c.png

3.3 方式三测试

1、调用localhost:8088/user3/get?id=11

90f0f37a-4cb5-11ee-a25d-92fbcf53809c.png

redis中缓存设置成功!

9105ecda-4cb5-11ee-a25d-92fbcf53809c.png

4. 常见报错

1、 ClassNotFoundException: com.alibaba.fastjson.JSON 解决:添加依赖


com.alibaba
fastjson
2.0.25

2、NoClassDefFoundError: redis/clients/jedis/UnifiedJedis 解决: 添加依赖


redis.clients
jedis
4.3.1

或者将jetcache版本降低至2.6.5以下

演示源码

https://gitee.com/wuhanxue/wu_study/tree/master/demo/jetcache-demo

审核编辑:汤梓红

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • JAVA
    +关注

    关注

    19

    文章

    2956

    浏览量

    104534
  • 缓存
    +关注

    关注

    1

    文章

    232

    浏览量

    26645
  • 开源
    +关注

    关注

    3

    文章

    3243

    浏览量

    42378
  • spring
    +关注

    关注

    0

    文章

    338

    浏览量

    14307
  • Redis
    +关注

    关注

    0

    文章

    371

    浏览量

    10843

原文标题:jetcache:阿里这款多级缓存框架一定要掌握

文章出处:【微信号:magedu-Linux,微信公众号:马哥Linux运维】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    java框架学习-Webwork2开发

    提供易于理解和维护的工具及开发框架,现在把我们java私塾做企业培训和日常教学用的webwork的资料共享出来,大家一起学习。第一部分 Webwork2简介WebWork是由OpenS
    发表于 09-29 14:15

    【南京】诚聘JAVA开发工程师

    猎头推荐职位:JAVA开发工程师工作职责:1. 了解互联网的技术发展、评估外部技术与解决方案;2. 能对系统进行性能优化,保证系统稳定、快速、安全运行;3. 能独自设计核心模块并完成代码编写
    发表于 07-05 14:45

    英创主板JAVA应用开发简介

    遇到问题,请参阅下载文档:《英创主板JAVA应用开发简介》中的“7、常见问题Q&A”。如果不能解决问题,请通过电话或者邮件联系我们。
    发表于 10-20 13:13

    java开源工具包-Jodd框架

    Jodd是一个Java工具包和微型框架,Jodd 工具包含一些实用的工具类和小型框架,增强了 JDK 提供很多强大的功能,可以帮助实现日常的开发任务,让代码更可靠;而Jodd
    发表于 03-19 16:13

    阿里巴巴开源的通用缓存访问框架JetCache介绍

    摘要: JetCache是由阿里巴巴开源的通用缓存访问框架,如果你对Spring Cache很熟悉的话,请一定花一点时间了解一下JetCache,它更好用。
    发表于 04-24 16:09

    JS应用开发框架组件

    JS应用开发框架组件简介目录接口使用相关仓简介JS应用开发框架,是OpenHarmony为
    发表于 04-23 18:05

    鸿蒙应用开发的JS UI框架如何实现高德地图的访问?

    鸿蒙应用,现在分为Java UI框架和Ark UI框架,其中JS UI开发框架Ark UI框架
    发表于 04-28 11:44

    基于AOP的智能Web缓存框架

    通过引入面向方面编程技术,提出一种新的智能Web缓存框架。描述该框架的组成结构与工作原理,对缓存设计时需要解决的透明性、一致性、替换算法和预取策略等主要问题进行讨
    发表于 04-11 09:04 4次下载

    Java 使用Redis缓存工具的详细解说

    本文是关于Java 使用Redis缓存工具的详细解说。详细步骤请看下文
    的头像 发表于 02-09 14:10 7862次阅读
    <b class='flag-5'>Java</b> 使用Redis<b class='flag-5'>缓存</b>工具的详细解说

    Java Web的开发前奏详细资料免费下载

    本文档的详细介绍的是Java Web的开发前奏详细资料免费下载主要内容包括了:1、Java Web开发简介,2、HTML、JavaScrip
    发表于 02-21 10:37 18次下载
    <b class='flag-5'>Java</b> Web的<b class='flag-5'>开发</b>前奏详细资料免费下载

    你想选择哪一种Java框架

    对于学Java的人来说,学习和了解框架是必修的。
    发表于 08-09 15:47 1268次阅读

    基于Java的接口快速开发框架

    作者丨编码砖家 来源丨http://adkx.net/w5m4s Part1简介 magic-api 是一个基于Java的接口快速开发框架,编写接口将通过magic-api提供的UI界
    的头像 发表于 09-10 09:36 3693次阅读

    基于Java的接口快速开发框架——magic-api

    magic-api 是一个基于Java的接口快速开发框架,编写接口将通过magic-api提供的UI界面完成,自动映射为HTTP接口,无需定义Controller、Service、Dao、Mapper、XML、VO等
    发表于 07-19 11:42 878次阅读
    基于<b class='flag-5'>Java</b>的接口快速<b class='flag-5'>开发</b><b class='flag-5'>框架</b>——magic-api

    Ehcache!这才是Java本地缓存之王!

    Java而言,其常用的缓存解决方案有很多,例如数据库缓存框架EhCache,分布式缓存Memcached等,这些
    的头像 发表于 07-29 11:21 1684次阅读
    Ehcache!这才是<b class='flag-5'>Java</b>本地<b class='flag-5'>缓存</b>之王!

    基于springboot和vue框架Java

    本文将详细介绍基于Spring Boot和Vue框架进行Java应用开发的实践。首先,将介绍Spring Boot和Vue框架的基本概念及其优点。然后,将详细介绍如何搭建Spring
    的头像 发表于 12-03 15:15 959次阅读