美文网首页
RedisTemplate 的泛型和序列化

RedisTemplate 的泛型和序列化

作者: 流芳不待人 | 来源:发表于2019-10-17 17:28 被阅读0次

1.通常用法

@Resource
RedisTemplate<String, String> redisTemplate;

public void doTest(){
  redisTemplate.opsForValue().setIfAbset("key", "val", Duration.ofSecend(100));
}

RedisTemplate<String, String> 表示操作的 key 和 val 都是String类型。

如果 val 是 Integer

@Resource
RedisTemplate<String, Integer> redisTemplate;

public void doTest(){
  redisTemplate.opsForValue().setIfAbset("key", 100, Duration.ofSecend(100));
}

这个就可能报错了 这是因为springboot访问redis时的序列化操作。

2.Serializer

springboot与Redis的交互是以二进制方式进行(byte[])。为了支持Java中的数据类型,就要对操作的对象(key,value,hashKey,hashValue...)做序列化操作。

redisTemplate 只为 key value hashKey hashValue 设置serializer

Springboot 提供了几个序列化的方法。

  • JdkSerializationRedisSerializer 默认
  • StringRedisSerializer
  • 其他 或 自定义

JdkSerializationRedisSerializer的序列化操作函数签名如下

public byte[] serialize(@Nullable Object object)

StringRedisSerializer序列化操作函数签名如下

public byte[] serialize(@Nullable String string)

因为 StringRedisSerializer 只支持 string类型 所以如果使用RedisTemplate<String, Integer>就会报错。

如果使用 JdkSerializationRedisSerializer 则不仅支持 RedisTemplate<String, Integer>同样能支持任意自定义类型 RedisTemplate<String, Person>

然而这种默认的序列化方式会导致redis中保存的key和value可读性较差,出现一些不可读的16进制字符

\xAC\xED\00\0x5t\x00

自定义序列化

JdkSerializationRedisSerializer虽然在redis中保存的数据是不可读的,但是操作起来很方便。可直接指定返回值的类型,免去了再次转换之繁琐

其实现原理是在redis中存储的数据里包含着数据类型。

有没有一种可能:在redis中数据不保存类型信息,通过为template指定value的类型,获取期望类型值

即:使用StringRedisSerializer,但能达到JdkSerializationRedisSerializer的效果。

比如:

@Resource
RedisTemplate<String,Integer> tplInt;
@Resource
RedisTemplate<String,Person> tplPerson;
public void testGet(){
  Integer x = tplInt.get("valOfX");
  Person p = tplPerson.get("valOfPerson");
}

难点在于:get()内部是否知道外部期望的类型是什么?

答案是:没有办法通过代码传入,只能从获返回的数据中获取。 原因如下:

redis返回的是 byte[] 类型,要用一个 serializer 做反序列化。

RedisTemplate中的 serializer 得是一个bean,即是一个实例化的对象。这个对象要实现 RedisSerializer<T> 接口,因此必定要绑定在一个固定类型上,如果是String就不能是Integer。所以无法根据需要传入。

  • StringRedisSerializer 实现的是 RedisSerializer<String>
  • JdkSerializationRedisSerializer 实现的是 RedisSerializer<Object>

后者为了兼容所有类型,所以设置为Object,反序列化后的数据是一个Object,这样就丢掉了原本的所有信息。所以如果要返回外部需要的类型,只能在序列化后做一次值的类型转换。本质逻辑类似如下:

public Object get(String key){
  //get data from redis
  return (Object)(new Person())
}
Person p = (Person)get();

如果为某个操作设置了专门的serializer,由于 template是单例的,其他线程也会受到影响。

其他坑

  • 除了 StringRedisSerializer ,其他对字符串0的处理,会导致redis的incr类指令不可用。(int 0 正常)

3.结论

选择1

RedisTemplate<String,String>将就用

选择2

springboot的序列化可以自定义,自己搭配。比如

//默认改成 StringRedisSerializer,key类的原样
template.setDefaultSerializer(new StringRedisSerializer());
//为value支持复杂类型
JdkSerializationRedisSerializer jdkSerializer = new JdkSerializationRedisSerializer();
template.setValueSerializer(jdkSerializer);
template.afterPropertiesSet();

这样 key、hashKey、hashValue可读的。value来支持复杂类型

4.参考

相关文章

网友评论

      本文标题:RedisTemplate 的泛型和序列化

      本文链接:https://www.haomeiwen.com/subject/eesvmctx.html