溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

在Redis中使用Template存儲(chǔ)緩存數(shù)據(jù)出現(xiàn)亂碼如何解決

發(fā)布時(shí)間:2021-03-02 16:36:28 來(lái)源:億速云 閱讀:565 作者:Leah 欄目:開發(fā)技術(shù)

這篇文章給大家介紹在Redis中使用Template存儲(chǔ)緩存數(shù)據(jù)出現(xiàn)亂碼如何解決,內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對(duì)大家能有所幫助。

前言

RedisTemplate是Spring對(duì)于Redis的封裝。

在Redis中使用Template存儲(chǔ)緩存數(shù)據(jù)出現(xiàn)亂碼如何解決

如上圖所示,RedisTemplate中定義了對(duì)5種數(shù)據(jù)結(jié)構(gòu)操作。

redisTemplate.opsForList();//操作list
redisTemplate.opsForValue();//操作字符串
redisTemplate.opsForCluster();//集群時(shí)使用
redisTemplate.opsForGeo();//地理位置時(shí)使用
redisTemplate.opsForHash();//操作hash
redisTemplate.opsForSet();//操作set
redisTemplate.opsForZSet();//操作有序set

與StringRedisTemplate的區(qū)別

StringRedisTemplate繼承RedisTemplate。

它們采用的序列化策略不同:

* StringRedisTemplate默認(rèn)采用的是String的序列化策略,保存的key和value都是采用此策略序列化保存的。

* RedisTemplate默認(rèn)采用的是JDK的序列化策略,保存的key和value都是采用此策略序列化保存的。

RedisTemplate和StringRedisTemplate它們存取的數(shù)據(jù)是相互獨(dú)立的。

解決辦法

上文已經(jīng)提及,在動(dòng)手的過(guò)程中,我采用的是RedisTemplate,在傳遞String類型的數(shù)據(jù)結(jié)構(gòu)后,查看緩存會(huì)發(fā)現(xiàn)數(shù)據(jù)亂碼現(xiàn)象。

在Redis中使用Template存儲(chǔ)緩存數(shù)據(jù)出現(xiàn)亂碼如何解決

這時(shí)候我們需要修改RedisTemplate的序列化策略。

RedisSerializer<String> stringSerializer = new StringRedisSerializer();
      redisTemplate.setKeySerializer(stringSerializer);
      redisTemplate.setValueSerializer(stringSerializer);
      redisTemplate.setHashKeySerializer(stringSerializer);
      redisTemplate.setHashValueSerializer(stringSerializer);

但是注意一點(diǎn),由于采用了String的序列化策略,所以只接受value值類型為String的參數(shù)。

如果像我一樣傳遞了Integer類型的參數(shù),直接使用toString()方法存入緩存。

ops.set("stock", redPacket.getStock().toString(),TIME_OUT, TimeUnit.SECONDS);

在Redis中使用Template存儲(chǔ)緩存數(shù)據(jù)出現(xiàn)亂碼如何解決

這樣就解決了亂碼問(wèn)題。

附:SpringBoot啟動(dòng)實(shí)例化配置

@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;
  }
}

補(bǔ)充:redis key和value的亂碼問(wèn)題解決,含日期轉(zhuǎn)化格式問(wèn)題

在Redis中使用Template存儲(chǔ)緩存數(shù)據(jù)出現(xiàn)亂碼如何解決

在項(xiàng)目中,遇到的問(wèn)題是redis的key和value出現(xiàn)的亂碼問(wèn)題:

而原本的內(nèi)容為下:

{
  "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"
  }
}

原因:

是因?yàn)楹蛂edis內(nèi)部的編碼協(xié)議出現(xiàn)了問(wèn)題,所以需要改進(jìn)。spring提供了一個(gè)優(yōu)化方案。

springboot的redisTemplate改進(jìn)。

@Component
@EnableRedisHttpSession(maxInactiveIntervalInSeconds = 3600)
public class RedisConfig {
  @Bean
  public RedisTemplate redisTemplate(RedisConnectionFactory factory){
    RedisTemplate redisTemplate = new RedisTemplate();
    redisTemplate.setConnectionFactory(factory);
    //首先解決key的序列化問(wèn)題
    StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();
    redisTemplate.setKeySerializer(stringRedisSerializer);
    //解決value的序列化問(wèn)題
    Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
    redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
    return redisTemplate;
  }
}

比之前好了,但是還有點(diǎn)小問(wèn)題,json的數(shù)據(jù)比以前多了,這是因?yàn)槿掌诘霓D(zhuǎn)化出現(xiàn)問(wèn)題,這塊的知識(shí)觸及盲區(qū),就先把解決方案寫下面,以后有時(shí)間在研究。

在Redis中使用Template存儲(chǔ)緩存數(shù)據(jù)出現(xiàn)亂碼如何解決

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);//轉(zhuǎn)成
  }
}
@Component
@EnableRedisHttpSession(maxInactiveIntervalInSeconds = 3600)
public class RedisConfig {
  @Bean
  public RedisTemplate redisTemplate(RedisConnectionFactory factory){
    RedisTemplate redisTemplate = new RedisTemplate();
    redisTemplate.setConnectionFactory(factory);
    //首先解決key的序列化問(wèn)題
    StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();
    redisTemplate.setKeySerializer(stringRedisSerializer);
    //解決value的序列化問(wèn)題
    Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
    redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
    //改進(jìn)日期轉(zhuǎn)化問(wèn)題
    ObjectMapper objectMapper = new ObjectMapper();
    SimpleModule simpleModule = new SimpleModule();
    simpleModule.addSerializer(DateTime.class,new JodaDateTimeJsonSerializer());
    simpleModule.addDeserializer(DateTime.class,new JodaDateTimeJsonDeserializer());
//解決反序列化問(wèn)題 objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
    objectMapper.registerModule(simpleModule);
    jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
    redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
    return redisTemplate;
  }
}

最后終于出現(xiàn)了預(yù)期的效果

在Redis中使用Template存儲(chǔ)緩存數(shù)據(jù)出現(xiàn)亂碼如何解決

關(guān)于在Redis中使用Template存儲(chǔ)緩存數(shù)據(jù)出現(xiàn)亂碼如何解決就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI