解決RedisTemplate存儲至緩存數據出現亂碼的情況
前言
RedisTemplate是Spring對於Redis的封裝。
如上圖所示,RedisTemplate中定義瞭對5種數據結構操作。
redisTemplate.opsForList();//操作list redisTemplate.opsForValue();//操作字符串 redisTemplate.opsForCluster();//集群時使用 redisTemplate.opsForGeo();//地理位置時使用 redisTemplate.opsForHash();//操作hash redisTemplate.opsForSet();//操作set redisTemplate.opsForZSet();//操作有序set
與StringRedisTemplate的區別
StringRedisTemplate繼承RedisTemplate。
它們采用的序列化策略不同:
* StringRedisTemplate默認采用的是String的序列化策略,保存的key和value都是采用此策略序列化保存的。
* RedisTemplate默認采用的是JDK的序列化策略,保存的key和value都是采用此策略序列化保存的。
RedisTemplate和StringRedisTemplate它們存取的數據是相互獨立的。
解決辦法
上文已經提及,在動手的過程中,我采用的是RedisTemplate,在傳遞String類型的數據結構後,查看緩存會發現數據亂碼現象。
這時候我們需要修改RedisTemplate的序列化策略。
RedisSerializer<String> stringSerializer = new StringRedisSerializer(); redisTemplate.setKeySerializer(stringSerializer); redisTemplate.setValueSerializer(stringSerializer); redisTemplate.setHashKeySerializer(stringSerializer); redisTemplate.setHashValueSerializer(stringSerializer);
但是註意一點,由於采用瞭String的序列化策略,所以隻接受value值類型為String的參數。
如果像我一樣傳遞瞭Integer類型的參數,直接使用toString()方法存入緩存。
ops.set("stock", redPacket.getStock().toString(),TIME_OUT, TimeUnit.SECONDS);
這樣就解決瞭亂碼問題。
附:SpringBoot啟動實例化配置
@Configuration public class RedisConfigurtion { @Autowired private RedisTemplate redisTemplate; @Bean public RedisTemplate<String, Object> stringSerializerRedisTemplate() { RedisSerializer<String> stringSerializer = new StringRedisSerializer(); redisTemplate.setKeySerializer(stringSerializer); redisTemplate.setValueSerializer(stringSerializer); redisTemplate.setHashKeySerializer(stringSerializer); redisTemplate.setHashValueSerializer(stringSerializer); return redisTemplate; } }
補充:redis key和value的亂碼問題解決,含日期轉化格式問題
在項目中,遇到的問題是redis的key和value出現的亂碼問題:
而原本的內容為下:
{ "status":"success", "data":{ "id":3, "title":"花林", "price":99, "stock":81, "description":"美女一隻", "sales":17, "imgUrl":"https://xiaolei1996.oss-cn-shanghai.aliyuncs.com/blog/title/we1.jpg", "promoStatus":2, "promoPrice":50, "promoId":1, "startDate":"2020-03-23 21:50:59" } }
原因:
是因為和redis內部的編碼協議出現瞭問題,所以需要改進。spring提供瞭一個優化方案。
springboot的redisTemplate改進。
@Component @EnableRedisHttpSession(maxInactiveIntervalInSeconds = 3600) public class RedisConfig { @Bean public RedisTemplate redisTemplate(RedisConnectionFactory factory){ RedisTemplate redisTemplate = new RedisTemplate(); redisTemplate.setConnectionFactory(factory); //首先解決key的序列化問題 StringRedisSerializer stringRedisSerializer = new StringRedisSerializer(); redisTemplate.setKeySerializer(stringRedisSerializer); //解決value的序列化問題 Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class); redisTemplate.setValueSerializer(jackson2JsonRedisSerializer); return redisTemplate; } }
比之前好瞭,但是還有點小問題,json的數據比以前多瞭,這是因為日期的轉化出現問題,這塊的知識觸及盲區,就先把解決方案寫下面,以後有時間在研究。
public class JodaDateTimeJsonSerializer extends JsonSerializer<DateTime> { @Override public void serialize(DateTime value, JsonGenerator gen, SerializerProvider serializers) throws IOException { gen.writeString(value.toString("yyyy-MM-dd HH:mm:ss")); } }
public class JodaDateTimeJsonDeserializer extends JsonDeserializer<DateTime> { @Override public DateTime deserialize(JsonParser p, DeserializationContext ctxt ) throws IOException, JsonProcessingException { String dateString= p.readValueAs(String.class); DateTimeFormatter dateTimeFormatter = DateTimeFormat.forPattern("yyyy-MM-dd HH:mm:ss"); return DateTime.parse(dateString,dateTimeFormatter);//轉成 } }
@Component @EnableRedisHttpSession(maxInactiveIntervalInSeconds = 3600) public class RedisConfig { @Bean public RedisTemplate redisTemplate(RedisConnectionFactory factory){ RedisTemplate redisTemplate = new RedisTemplate(); redisTemplate.setConnectionFactory(factory); //首先解決key的序列化問題 StringRedisSerializer stringRedisSerializer = new StringRedisSerializer(); redisTemplate.setKeySerializer(stringRedisSerializer); //解決value的序列化問題 Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class); redisTemplate.setValueSerializer(jackson2JsonRedisSerializer); //改進日期轉化問題 ObjectMapper objectMapper = new ObjectMapper(); SimpleModule simpleModule = new SimpleModule(); simpleModule.addSerializer(DateTime.class,new JodaDateTimeJsonSerializer()); simpleModule.addDeserializer(DateTime.class,new JodaDateTimeJsonDeserializer()); //解決反序列化問題 objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL); objectMapper.registerModule(simpleModule); jackson2JsonRedisSerializer.setObjectMapper(objectMapper); redisTemplate.setValueSerializer(jackson2JsonRedisSerializer); return redisTemplate; } }
最後終於出現瞭預期的效果
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。如有錯誤或未考慮完全的地方,望不吝賜教。
推薦閱讀:
- 解決SpringBoot下Redis序列化亂碼的問題
- Redis序列化存儲及日期格式的問題處理
- 使用@Autowired 註入RedisTemplate報錯的問題及解決
- springboot2.5.0和redis整合配置詳解
- SpringBoot結合Redis實現序列化的方法詳解