Spring使用redis遇到的問題及解決方案
本人在spring中使用redis作為緩存時(shí),遇到兩個(gè)坑,現(xiàn)在記錄如下,算是作為自己的備忘吧,文筆不好,望大家見諒;
一、配置文件
<!-- 加載Properties文件 -->
<bean id="configurer" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="locations">
<list>
<value>classpath*:config.properties</value>
</list>
</property>
<property name="ignoreUnresolvablePlaceholders" value="true"/>
</bean>
<!-- 配置JedisPoolConfig實(shí)例 -->
<bean id="jedisPoolConfig" class="redis.clients.jedis.JedisPoolConfig">
<!--最大空閑數(shù),數(shù)據(jù)庫連接的最大空閑時(shí)間。超過空閑時(shí)間,數(shù)據(jù)庫連接將被標(biāo)記為不可用,然后被釋放。設(shè)為0表示無限制-->
<property name="maxIdle" value="300" />
<!--連接池的最大數(shù)據(jù)庫連接數(shù)。設(shè)為0表示無限制-->
<property name="maxTotal" value="600" />
<!--最大建立連接等待時(shí)間。如果超過此時(shí)間將接到異常。設(shè)為-1表示無限制-->
<property name="maxWaitMillis" value="1000" />
<!--在borrow一個(gè)jedis實(shí)例時(shí),是否提前進(jìn)行alidate操作;如果為true,則得到的jedis實(shí)例均是可用的-->
<property name="testOnBorrow" value="true" />
<!--每次逐出檢查時(shí) 逐出的最大數(shù)目 如果為負(fù)數(shù)就是 : 1/abs(n), 默認(rèn)3-->
<property name="numTestsPerEvictionRun" value="3"/>
<!--逐出連接的最小空閑時(shí)間 默認(rèn)1800000毫秒(30分鐘)-->
<property name="minEvictableIdleTimeMillis" value="300000"/>
<!--逐出掃描的時(shí)間間隔(毫秒) 如果為負(fù)數(shù),則不運(yùn)行逐出線程, 默認(rèn)-1-->
<property name="timeBetweenEvictionRunsMillis" value="60000"/>
<!--在空閑時(shí)檢查有效性, 默認(rèn)false -->
<property name="testWhileIdle" value="true" />
</bean>
<!-- 配置JedisConnectionFactory,類似于數(shù)據(jù)庫的連接池 -->
<bean id="jedisConnectionFactory" class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory">
<property name="hostName" value="${redis.host}"></property>
<property name="port" value="${redis.port}"></property>
<property name="password" value="${redis.password}"></property>
<property name="database" value="${redis.dbIndex}"></property>
<property name="poolConfig" ref="jedisPoolConfig"></property>
<property name="timeout" value="100000"></property>
</bean>
<!-- 配置RedisTemplate -->
<bean id="redisTemplate" class="org.springframework.data.redis.core.RedisTemplate">
<property name="connectionFactory" ref="jedisConnectionFactory"></property>
<!--如果不配置Serializer,那么存儲(chǔ)的時(shí)候缺省使用String,如果用User類型存儲(chǔ),那么會(huì)提示錯(cuò)誤User can't cast to String??! -->
<property name="keySerializer" >
<bean class="org.springframework.data.redis.serializer.StringRedisSerializer" />
</property>
<property name="valueSerializer" >
<bean class="org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer" />
</property>
<property name="hashKeySerializer">
<bean class="org.springframework.data.redis.serializer.StringRedisSerializer"/>
</property>
<property name="hashValueSerializer">
<bean class="org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer"/>
</property>
<!--開啟事務(wù) -->
<property name="enableTransactionSupport" value="true"></property>
</bean>
注:可以看到我redis的配置中,對(duì)于hash的key使用的是StringRedisSerializer序列化,而對(duì)于value使用的是GenericJackson2JsonRedisSerializer序列化。
二、坑一
// 源代碼
HashOperations ho = redisTemplate.opsForHash();
Boolean flag = ho.hasKey(“key”, "key1");
if (flag) {
// 處理邏輯
}
上面這段代碼,當(dāng)redis中值不存在時(shí),按照官方的說明文檔,應(yīng)該返回false。但是我使用的時(shí)候,hasKey方法時(shí)而返回的是null,時(shí)而返回的false,導(dǎo)致空指針異常。搞了半天我也沒搞明白為啥返回false,最后沒辦法我妥協(xié)了(大神如果知道可以回復(fù)我)。
// 妥協(xié)后代碼
HashOperations ho = redisTemplate.opsForHash();
Boolean flag = ho.hasKey(“key”, "key1");
if (flag != null && flag) {
// 處理邏輯
}
三、坑二
// 將Map放到redis的hash中
public void putRedisHash1(){
HashOperations ho = redisTemplate.opsForHash();
Map<String, Boolean> tempMap = new HashMap<String, Boolean>(3){{
put("isRegistered",false);
put("isWeChat",false);
put("isAliPay",false);
}};
ho.put("key", "key1", tempMap);
}
// 將Map放到redis的hash中
public void putRedisHash2(){
HashOperations ho = redisTemplate.opsForHash();
Map<String, Boolean> tempMap = new HashMap<String, Boolean>(3);
tempMap.put("isRegistered",false);
tempMap.put("isWeChat",false);
tempMap.put("isAliPay",false);
ho.put("key", "key1", tempMap);
}
這兩個(gè)方法的功能是一樣的,但是第一個(gè)方法是在Map初始化同時(shí)將值放入其中,第二個(gè)方法是Map初始化以后將值放入其中,執(zhí)行結(jié)果是一樣,但是在redis中的存儲(chǔ)形式完全不同。由于Hash的value使用的是GenericJackson2JsonRedisSerializer序列化,所以為了反序列化方便,它會(huì)存儲(chǔ)
@class“”這個(gè)字段,由于Map初始化時(shí)機(jī)不同,導(dǎo)致相同內(nèi)容在redis中@class內(nèi)容不一致,為了使用方便,推薦第二種,即在Map初始化以后將值放入其中。

以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
偵聽消息隊(duì)列的Message Listener類示例詳解
Spring AMQP 是基于 Spring 框架的AMQP消息解決方案,提供模板化的發(fā)送和接收消息的抽象層,提供基于消息驅(qū)動(dòng)的 POJO的消息監(jiān)聽等,簡(jiǎn)化了我們對(duì)于RabbitMQ相關(guān)程序的開發(fā),本文給大家介紹偵聽消息隊(duì)列的Message Listener類,感興趣的朋友一起看看吧2023-12-12
SpringBoot項(xiàng)目開發(fā)中常用的依賴
這篇文章主要介紹了SpringBoot項(xiàng)目開發(fā)中常用的依賴詳解,本文通過示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-06-06
Springboot使用Logback實(shí)現(xiàn)日志配置與異常記錄
默認(rèn)情況下,SpringBoot內(nèi)部使用logback作為系統(tǒng)日志實(shí)現(xiàn)的框架,將日志輸出到控制臺(tái),不會(huì)寫到日志文件。本篇文章主要講解下如何自定義logabck.xml以及對(duì)logback文件中配置做一個(gè)詳解,需要的可以參考一下2022-11-11
springboot Controller直接返回String類型帶來的亂碼問題及解決
文章介紹了在Spring Boot中,當(dāng)Controller直接返回String類型時(shí)可能出現(xiàn)的亂碼問題,并提供了解決辦法,通過在`application.yaml`中設(shè)置請(qǐng)求和響應(yīng)的編碼格式,并在自定義配置類中進(jìn)行配置,可以有效解決這一問題2024-11-11

