redisson緩存序列化幾枚坑
1、返回值為MapT, K> 的方法增加@Cacheable后,T和K被類型擦出了,為啥?
redisson結(jié)合Spring使用時,會有RedissonSpringCacheManager,將redissonClient自動注入,另外還有codec的概念,即序列化和反序列化,可以查看實(shí)現(xiàn)類,就幾種實(shí)現(xiàn),假設(shè)我們使用org.redisson.codec.JsonJacksonCodec,可以看到,decode中,僅一個Object.class,即范型信息并未帶入,故出現(xiàn)了問題
2、對于匿名內(nèi)部類的濫用導(dǎo)致反序列化失敗
你會想,匿名內(nèi)部類有什么影響?
那么跟著我看下我們時常會寫的一種Map寫法:
MapString, Object> map = new HashMap(){{put("mykey", "test");}};
這種方式有什么問題呢,這就涉及到匿名內(nèi)部類聲明方式在實(shí)際編譯時是如何存在于class文件中的
...
$1 extends HashMap{
...
}
...
也就是新生成了一個匿名類型,而這個類型在反序列化時是沒辦法找到構(gòu)造函數(shù)的,故而是有問題的。
按上面寫法后,序列化時,存儲的是xxx$1這個匿名類型,所以反序列化也就失敗了。
redis的坑(序列化、scan)
最近做的一個項(xiàng)目用到redis,需要使用redis對數(shù)據(jù)進(jìn)行緩存,用戶的動作也會更新redis中的數(shù)據(jù),為了方便管理,采用了hash的方式。神坑就此開始。
最開始是序列化的坑
使用包裝的ByteArrayRedisTemplate時,對象存入redis之后,rdm可以查看到,但是程序里面取出來是亂碼,使用原生的RedisTemplate就不會出現(xiàn)這個問題,后來發(fā)現(xiàn)是對象包裝的問題,原生的RedisTemplate中支持將value設(shè)置為對象,但是包裝的ByteArrayRedisTemplate只能用byte[],所以我這邊先把對象轉(zhuǎn)為json,然后json轉(zhuǎn)為byte[],再寫入redis,取數(shù)據(jù)的時候,查redis的結(jié)果是byte[],然后轉(zhuǎn)為json,再轉(zhuǎn)為對象,就沒問題了。
但是?。?!不知道什么原因,這樣做之后rdm中查不到這個key了,可能是redis版本和rdm版本不兼容的問題,這個有待驗(yàn)證。你看到的一切不一定存在,你看不到的也不一定不存在,當(dāng)個碼農(nóng)還要思考這些哲學(xué)問題。。。
還有一個坑
spring整合的redis是不支持scan指令的,而且不只是scan指令,基本上所有摟全量的指令都被禁止,當(dāng)然,像keys之類的指令還是能用,但是在生產(chǎn)環(huán)境下千萬不要使用,因?yàn)楹苋菀鬃枞瑯I(yè)務(wù)動不動就停幾秒,很尷尬。而且現(xiàn)在大部分在生產(chǎn)環(huán)境下使用的redis都是用codis包裝的,codis更絕,直接禁止使用那些指令,同志們可以自己動手搜一下,被禁止的指令還是挺多得,我第一次看還以為自己看錯了,尼瑪禁了一大半,摟全量的指令全部被禁。不過這樣做的好處就是數(shù)據(jù)安全,使用scan指令的漏洞撈數(shù)據(jù)的軟件也不在少數(shù)。
最后項(xiàng)目只能放棄使用redis了,因?yàn)槲冶仨毜脫?。。。通過這個事件也懂得了,代碼開發(fā)一定要一邊開發(fā)一邊測試(自己測試),不然有的坑,掉進(jìn)去了都不知道,還在屁顛屁顛的往里刨,最后把自己埋了。
以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。
您可能感興趣的文章:- 詳解Redis緩存穿透/擊穿/雪崩原理及其解決方案
- java若依框架集成redis緩存詳解
- Redis使用元素刪除的布隆過濾器來解決緩存穿透問題
- springboot使用Redis作緩存使用入門教程
- 淺談Redis 緩存的三大問題及其解決方案
- 淺談java如何實(shí)現(xiàn)Redis的LRU緩存機(jī)制
- 在項(xiàng)目中使用redis做緩存的一些思路