惊讶!缓存刚Put再Get竟然获取不到?

最近一直在老家远程办公,微信忽然响了下,有同事说遇到了一个奇怪的问题,让我帮忙看下。java

现象就是标题所说的缓存获取不到的问题,我一听感受这个问题挺有意思的,决定一探究竟。git

下面给出部分代码还原下案发现场:github

@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秒,确定不是刚保存就过时了的缘由。bash

而后去Redis中查看到底有没有Put进去,发现数据在Redis中已经存在了,证实插入没问题。微信

图片

只有使出终极必杀器了,那就是debug源码。app

经过get方法一直往下看,最终到了RedisCache里面。框架

图片

而后在这里打个断点,看看到底有没有获取到Redis中的值,惊讶的发现,值是获取到了的,以下:函数

图片

纳尼,这是什么操做。摸了摸我还没秃顶的后脑勺,我锁定了下面这行代码:this

CacheValueHolder<V> holder = (CacheValueHolder)this.valueDecoder.apply(bytes);
复制代码

猜想应该是解码的时候出问题了,而后找到了对应的解码的代码,用的是kryo框架。

图片

终于在最后一步解码的时候发现了错误,守得云开见月明啊!

图片

错误告诉咱们ArrayList缺乏构造函数呀,请注意是Arrays里面的ArrayList。吓得我赶忙看下代码,果然是Arrays.asList()构造的参数。

图片

*解决办法天然就很简单了,直接用 *java.util.ArrayList便可。

最后想说的是解决问题最重要的是方式和技巧。写这篇文章的目的也是但愿你们在遇到问题的时候不要局限于表面,能够往深一点去探索。

感兴趣的能够关注下个人微信公众号 猿天地,更多技术文章第一时间阅读。个人GitHub也有一些开源的代码 github.com/yinjihuan

相关文章
相关标签/搜索