目录
痛点:
spring-data-redis中redisTemplate默认使用JDK的序列化策略,会出现两个问题
- 使用redis-cli查看数据时,携带很多字符,不易查看
- JDK Serializer太费资源
解决方案: 使用Jackson serializer替代JDK Serializer;
StringRedisSerializer结合GenericJackson2JsonRedisSerializer
1. spring-data-redis支持的序列化策略
spring-data-redis默认采用的序列化策略有两种,一种是String的序列化策略,一种是JDK的序列化策略。
参考博客 https://blog.csdn.net/pcwblover008/article/details/79915205
redisTemplate可自定义各种key和各种value的序列化方式:
- defaultSerializer 默认序列化策略
- key 普通key,非hash
- value 普通value,非hash
- hashKey hash的filed
- hashValue hash的value
spring-data-redis的序列化类有下面这几个:
- GenericToStringSerializer: 可以将任何对象泛化为字符串并序列化
- Jackson2JsonRedisSerializer: 跟JacksonJsonRedisSerializer实际上是一样的
- JacksonJsonRedisSerializer: 序列化object对象为json字符串
- JdkSerializationRedisSerializer: 序列化java对象(被序列化的对象必须实现Serializable接口)
- StringRedisSerializer: 简单的字符串序列化
- GenericToStringSerializer:类似StringRedisSerializer的字符串序列化
- GenericJackson2JsonRedisSerializer:类似Jackson2JsonRedisSerializer,但使用时构造函数不用特定的类参考以上序列化,自定义序列化类;
博客https://stackoverflow.com/questions/13215024/weird-redis-key-with-spring-data-jedis?answertab=votes#tab-top中,使用了Jackson2JsonRedisSerializer, 需要在配置文件中, 配置一个类,用于反序列化. 这是致命缺陷!!
所以, 我们推荐使用GenericJackson2JsonRedisSerializer
<bean id="userJsonRedisSerializer" class="org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer">
<constructor-arg type="java.lang.Class" value="com.mycompany.redis.domain.User"/>
</bean>
StringRedisTemplate默认采用的是String的序列化策略,保存的key和value都是采用此策略序列化保存的。StringRedisSerializer
RedisTemplate默认采用的是JDK的序列化策略,保存的key和value都是采用此策略序列化保存的。JdkSerializationRedisSerializer
key和hashKey: 推荐使用 StringRedisSerializer: 简单的字符串序列化
hashValue: 推荐使用 GenericJackson2JsonRedisSerializer:类似Jackson2JsonRedisSerializer,但使用时构造函数不用特定的类
2. 修改spring配置文件
参考博客 https://www.cnblogs.com/grey-wolf/p/7910232.html
策略: defaultSerializer, 设置为GenericJackson2JsonRedisSerializer;
所有的key(key和hashKey), 设置为StringRedisSerializer
SpringDataRedisDemo工程使用的配置文件信息:
<context:property-placeholder location="classpath*:properties/*.properties" />
<!-- redis 相关配置 -->
<bean id="poolConfig" class="redis.clients.jedis.JedisPoolConfig">
<property name="maxIdle" value="${redis.maxIdle}" />
<property name="maxWaitMillis" value="${redis.maxWait}" />
<property name="testOnBorrow" value="${redis.testOnBorrow}" />
</bean>
<bean id="JedisConnectionFactory"
class="org.springframework.data.redis.connection.jedis.JedisConnectionFactory"
p:host-name="${redis.host}" p:port="${redis.port}" p:password="${redis.pass}"
p:pool-config-ref="poolConfig" />
<bean id="redisTemplate" class="org.springframework.data.redis.core.RedisTemplate">
<property name="connectionFactory" ref="JedisConnectionFactory" />
<!-- 配置默认的序列化策略,非hash的value和hashValue都有效-->
<property name="defaultSerializer" ref="genericJackson2JsonRedisSerializer">
</property>
<!-- 配置redis的key的序列化Serializer方式,使5中类型,key都以String类型进行序列化 -->
<!-- Hash类型,用Jedis存储key为brandList,但是在redis-cli中查看key却是 "\xac\xed\x00\x05t\x00\tbrandList" -->
<property name="keySerializer" ref="stringRedisSerializer" />
<property name="hashKeySerializer" ref="stringRedisSerializer" />
</bean>
<!-- 配置key的序列化方式,使用String类型进行序列化 -->
<bean id="stringRedisSerializer"
class="org.springframework.data.redis.serializer.StringRedisSerializer" />
<!-- 配置hashValue的序列化方式,使用Jackson serializer,将hashValue以json串的形式存储到redis中 -->
<bean id="genericJackson2JsonRedisSerializer"
class="org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer">
</bean>
测试: hash结构
1. key为"namehash", filed为"e", 添加value为数组{1,2,3}
2. key为"namehash", filed为"f", 添加value为列表["齐一","牛二"]
3. key为"namehash", filed为"g", 添加value为散列{"1":"胡三","2":"虎四","3":"marin"}
结果是:
注意: 如果要redis-cli显示中文,需要启动redis_cli的时候,在后面加上 --raw