惊讶!缓存刚Put再Get居然获取不到?
最近一直在老家远程办公,微信突然响了下,有同事说遇到了一个奇怪的问题,让我帮忙看下。
现象就是标题所说的缓存获取不到的问题,我一听感觉这个问题挺有意思的,决定一探究竟。
下面给出部分代码还原下案发现场:
@CreateCache(name = "demo", expire = 600)
private Cache<String, ThirdPartyEventResponse> cache;
@Test
public void test() {
ThirdPartyEventResponse eventResponse = new ThirdPartyEventResponse();
eventResponse.setTicketCategories(Arrays.asList(ticketCategoryResponse));
// 省略 .....
// 添加
cache.put(DisChannelType.PIAONIU.getValue(), eventResponse);
// 获取
ThirdPartyEventResponse resp = cache.get(DisChannelType.PIAONIU.getValue());
}
Put之后马上Get,居然获取不到值。这就有点匪夷所思了,我们来好好排查下。
首先过期时间为600秒,肯定不是刚保存就过期了的原因。
然后去Redis中查看到底有没有Put进去,发现数据在Redis中已经存在了,证明插入没问题。
只有使出终极必杀器了,那就是debug源码。
通过get方法一直往下看,最终到了RedisCache里面。
然后在这里打个断点,看看到底有没有获取到Redis中的值,惊讶的发现,值是获取到了的,如下:
纳尼,这是什么操作。摸了摸我还没秃顶的后脑勺,我锁定了下面这行代码:
CacheValueHolder<V> holder = (CacheValueHolder)this.valueDecoder.apply(bytes);
猜测应该是解码的时候出问题了,然后找到了对应的解码的代码,用的是kryo框架。
终于在最后一步解码的时候发现了错误,守得云开见月明啊!
错误告诉我们ArrayList缺少构造函数呀,请注意是Arrays里面的ArrayList。吓得我赶紧看下代码,果真是Arrays.asList()构造的参数。
*解决办法自然就很简单了,直接用 *java.util.ArrayList即可。
最后想说的是解决问题最重要的是方式和技巧。写这篇文章的目的也是希望大家在遇到问题的时候不要局限于表面,可以往深一点去探索。
欢迎大家关注我的微信公众号‘猿天地’,
猿天地由《Spring Cloud微服务-全栈技术与案例解析》作者尹吉欢创建。
JAVA方向为主(Spring Boot, Spring Cloud,Sharding JDBC,Elastic Job,Mongodb等)。
技术之路很漫长,我们一起前行吧。
作者代码分享GitHub地址:https://github.com/yinjihuan ,
个人网站:http://cxytiandi.com/course ,
有任何问题或者合作请联系我。微信号:jihuan900
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 地球OL攻略 —— 某应届生求职总结
· 提示词工程——AI应用必不可少的技术
· Open-Sora 2.0 重磅开源!
· 周边上新:园子的第一款马克杯温暖上架