解決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。如有錯誤或未考慮完全的地方,望不吝賜教。

推薦閱讀: