0. 引言
在实际应用中,并不是单一的使用本地缓存或者redis,更多是组合使用来满足不同的业务场景,于是如何优雅的组合本地缓存和远程缓存就成了我们要研究的问题,而这一点,阿里开源的jetcache组件帮我们实现了
1. jetcache简介
jetcache是阿里开源的基于java开发的缓存框架,支持多种缓存类型:本地缓存、分布式缓存、多级缓存。能够满足不同业务场景的缓存需求。
jetcache具有上手简单、性能高效、拓展性强的特点。支持缓存预热 、缓存key前缀等功能。结合spring-cache使用,可以实现十分优雅的缓存类型切换
2. jetcache使用
1、引入依赖,这里我们使用sringboot项目框架,同时使用redis作为远程缓存。于是我们引入jetcache-starter-redis依赖,这里我的springboot版本为2.6.13
如果是非springboot项目可以参考官网说明配置
com.alicp.jetcache jetcache-starter-redis 2.7.0 redis.clients jedis 4.3.1
对应的版本说明如下:springboot与jetcache版本关系
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
更详细的参数配置
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 publicCache getUserCache(){ 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
因为配置的是远程缓存,在redis中也能看到对应的key
2、访问localhost:8088/user/getLocal?id=1,这个方法是从本地缓存获取的,现在只有远程缓存上有数据,我们调用发现缓存数据还是拿到了,这说明当我们在配置文件中配置了本地缓存和远程缓存后,方式一中本地缓存和远程缓存会自动相互调用
比如本地缓存有这个key,redis中没有,通过远程缓存方式访问时,会先从redis获取,如果没有会自动获取本地缓存,但是数据还是存储在本地缓存,并不会同步到redis上,这样更加灵活的实现了多级缓存架构
3.2 方式二测试
1、再测试下CreateCache的形式:localhost:8088/user2/get?id=4
正常获取了,并且redis中也有了对应的值
而当我们把缓存方式更改为LOCAL后,再访问localhost:8088/user2/get?id=5
@CreateCache(name="userCache:",expire=3600,timeUnit=TimeUnit.SECONDS,cacheType=CacheType.LOCAL)
会发现redis中就没有对应缓存了,只在本地缓存存在,说明我们指定本地缓存的形式成功了
3.3 方式三测试
1、调用localhost:8088/user3/get?id=11
redis中缓存设置成功!
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运维】欢迎添加关注!文章转载请注明出处。
发布评论请先 登录
相关推荐
评论