springboot2 redis緩存配置Jackson2JsonRedisSerializer和使用object.class時(shí)反序列化出現(xiàn)LinkedHashMap cannot be cas...

前言

最近在學(xué)習(xí)springboot的整合視頻,在整合redis作為cache時(shí)序列化為json格式時(shí),由于視頻上講述的是springboot1的修改方式毛仪,而springboot2在這部分有了較大的修改排嫌。在網(wǎng)上我查到的基本上大多是自己配置一個CacheManager來替換springboot默認(rèn)的。但這樣做就相當(dāng)于把springboot默認(rèn)的配置全部替換掉了仿吞。包括CacheManagerCustomizers和CacheProperties滑频。但我只是想修改成json格式序列化而已,沒必要改動其他的內(nèi)容唤冈。

查看源碼

最初我準(zhǔn)備把官方的RedisCacheConfiguration中的RedisCacheManager配置源碼全部復(fù)制過來峡迷,只修改序列化規(guī)則。在修改過程中發(fā)現(xiàn)源碼中ObjectProvider這個接口多次出現(xiàn)你虹,在經(jīng)過百度后大致認(rèn)為在springboot自動注入?yún)?shù)時(shí)绘搞,如果IOC容器中不存在這個bean是會報(bào)錯的。使用ObjectProvider則允許參數(shù)為空傅物。

查看cacheManager()函數(shù)

    @Bean
    RedisCacheManager cacheManager(CacheProperties cacheProperties, CacheManagerCustomizers cacheManagerCustomizers,
            ObjectProvider<org.springframework.data.redis.cache.RedisCacheConfiguration> redisCacheConfiguration,
            ObjectProvider<RedisCacheManagerBuilderCustomizer> redisCacheManagerBuilderCustomizers,
            RedisConnectionFactory redisConnectionFactory, ResourceLoader resourceLoader) {
        //創(chuàng)建RedisCacheManagerBuilder
        RedisCacheManagerBuilder builder = RedisCacheManager.builder(redisConnectionFactory).cacheDefaults(
                determineConfiguration(cacheProperties, redisCacheConfiguration, resourceLoader.getClassLoader()));
        List<String> cacheNames = cacheProperties.getCacheNames();
        if (!cacheNames.isEmpty()) {
            builder.initialCacheNames(new LinkedHashSet<>(cacheNames));
        }
        redisCacheManagerBuilderCustomizers.orderedStream().forEach((customizer) -> customizer.customize(builder));
        return cacheManagerCustomizers.customize(builder.build());
    }

首先可以確定的是官方默認(rèn)的cacheManager便是由這個函數(shù)創(chuàng)建的看杭。在創(chuàng)建RedisCacheManagerBuilder的時(shí)候設(shè)置的cacheDefaults是通過determineConfiguration()這個函數(shù)得到的。

查看determineConfiguration()函數(shù)

    private org.springframework.data.redis.cache.RedisCacheConfiguration determineConfiguration(
            CacheProperties cacheProperties,
            ObjectProvider<org.springframework.data.redis.cache.RedisCacheConfiguration> redisCacheConfiguration,
            ClassLoader classLoader) {
        return redisCacheConfiguration.getIfAvailable(() -> createConfiguration(cacheProperties, classLoader));
    }

可以看到這里的redisCacheConfiguration就是一個ObjectProvider接口挟伙,而這個函數(shù)本體只有一行內(nèi)容

return redisCacheConfiguration.getIfAvailable(() -> createConfiguration(cacheProperties, classLoader));

而這個返回值是由ObjectProvider接口的getIfAvailable()函數(shù)產(chǎn)生的楼雹。點(diǎn)開getIfAvailable()函數(shù)

    default T getIfAvailable(Supplier<T> defaultSupplier) throws BeansException {
        T dependency = this.getIfAvailable();
        return dependency != null ? dependency : defaultSupplier.get();
    }

通過內(nèi)容大致推測為如果自動注入時(shí)容器中本身就存在這個泛型的bean則返回這個bean,否則則返回參數(shù)生成的bean尖阔。而參數(shù)() -> createConfiguration(cacheProperties, classLoader)調(diào)用的createConfiguration則是springboot默認(rèn)創(chuàng)建的RedisCacheConfiguration贮缅。而序列化等配置信息就是通過這個createConfiguration來調(diào)整的。

推測

那么是否我在config中自定義一個RedisCacheConfiguration放入容器即可修改序列化方式呢介却?

于是我將createConfiguration的代碼復(fù)制并進(jìn)行修改

@Configuration
public class MyRedisConfig {
    @Bean
    RedisCacheConfiguration myConfiguration(
            CacheProperties cacheProperties) {
        CacheProperties.Redis redisProperties = cacheProperties.getRedis();
        RedisCacheConfiguration config = RedisCacheConfiguration
                .defaultCacheConfig();
        //將原本的JdkSerializationRedisSerializer修改為Jackson2JsonRedisSerializer
        Jackson2JsonRedisSerializer serializer = new Jackson2JsonRedisSerializer(Object.class);
        config = config.serializeValuesWith(
                RedisSerializationContext.SerializationPair.fromSerializer(serializer));
        if (redisProperties.getTimeToLive() != null) {
            config = config.entryTtl(redisProperties.getTimeToLive());
        }
        if (redisProperties.getKeyPrefix() != null) {
            config = config.prefixCacheNameWith(redisProperties.getKeyPrefix());
        }
        if (!redisProperties.isCacheNullValues()) {
            config = config.disableCachingNullValues();
        }
        if (!redisProperties.isUseKeyPrefix()) {
            config = config.disableKeyPrefix();
        }
        return config;
    }
}

啟動項(xiàng)目

進(jìn)行查詢谴供,查看redis


果然成功了!齿坷!

踩坑

但是桂肌,當(dāng)我再次刷新時(shí)發(fā)生了錯誤LinkedHashMap cannot be cast to。再次面向百度編程后了解到永淌,由于我定義的是object.class作為轉(zhuǎn)換崎场。并沒有給定具體的對象,所以在反序列化時(shí)遂蛀,被轉(zhuǎn)換成了LinkedHashMap谭跨。設(shè)置屬性DefaultTyping.NON_FINAL后問題解決了。

        Jackson2JsonRedisSerializer serializer = new Jackson2JsonRedisSerializer(Object.class);
        //序列化時(shí)添加對象信息
        ObjectMapper objectMapper = new ObjectMapper();
        objectMapper.activateDefaultTyping(objectMapper.getPolymorphicTypeValidator(), ObjectMapper.DefaultTyping.NON_FINAL);
        serializer.setObjectMapper(objectMapper);
        config = config.serializeValuesWith(
                RedisSerializationContext.SerializationPair.fromSerializer(serializer));

清空redis并重啟項(xiàng)目

再次進(jìn)行查詢,并查看redis

數(shù)據(jù)變成了這種格式螃宙。

刷新多次進(jìn)行查詢蛮瞄。除了第一次經(jīng)過了數(shù)據(jù)庫后其他幾次都是通過緩存,問題解決谆扎。


?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末挂捅,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子堂湖,更是在濱河造成了極大的恐慌籍凝,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,544評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件苗缩,死亡現(xiàn)場離奇詭異,居然都是意外死亡声诸,警方通過查閱死者的電腦和手機(jī)酱讶,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,430評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來彼乌,“玉大人泻肯,你說我怎么就攤上這事∥空眨” “怎么了灶挟?”我有些...
    開封第一講書人閱讀 162,764評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長毒租。 經(jīng)常有香客問我稚铣,道長,這世上最難降的妖魔是什么墅垮? 我笑而不...
    開封第一講書人閱讀 58,193評論 1 292
  • 正文 為了忘掉前任惕医,我火速辦了婚禮,結(jié)果婚禮上算色,老公的妹妹穿的比我還像新娘抬伺。我一直安慰自己,他們只是感情好灾梦,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,216評論 6 388
  • 文/花漫 我一把揭開白布峡钓。 她就那樣靜靜地躺著,像睡著了一般若河。 火紅的嫁衣襯著肌膚如雪能岩。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,182評論 1 299
  • 那天萧福,我揣著相機(jī)與錄音捧灰,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛毛俏,可吹牛的內(nèi)容都是我干的炭庙。 我是一名探鬼主播,決...
    沈念sama閱讀 40,063評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼煌寇,長吁一口氣:“原來是場噩夢啊……” “哼焕蹄!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起阀溶,我...
    開封第一講書人閱讀 38,917評論 0 274
  • 序言:老撾萬榮一對情侶失蹤腻脏,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后银锻,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體永品,經(jīng)...
    沈念sama閱讀 45,329評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,543評論 2 332
  • 正文 我和宋清朗相戀三年击纬,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了鼎姐。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,722評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡更振,死狀恐怖炕桨,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情肯腕,我是刑警寧澤献宫,帶...
    沈念sama閱讀 35,425評論 5 343
  • 正文 年R本政府宣布,位于F島的核電站实撒,受9級特大地震影響姊途,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜知态,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,019評論 3 326
  • 文/蒙蒙 一吭净、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧肴甸,春花似錦寂殉、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,671評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至庶柿,卻和暖如春村怪,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背浮庐。 一陣腳步聲響...
    開封第一講書人閱讀 32,825評論 1 269
  • 我被黑心中介騙來泰國打工甚负, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留柬焕,地道東北人。 一個月前我還...
    沈念sama閱讀 47,729評論 2 368
  • 正文 我出身青樓梭域,卻偏偏與公主長得像斑举,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子病涨,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,614評論 2 353