前言
本文仅对使用RedisTemplate操作list(对象)集合过程中遇到的一些问题,进行记录。
SpringBoot版本:2.2.2.RELEASE
Redis版本:3.2.1.0.0
导入依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
SpringBoot为我们操作Redis提供了两个API,StringRedisTemplate(默认采用的是StringRedisSerializer序列化策略)、RedisTemplate<String,Object>(默认采用的是JdkSerializationRedisSerializer序列化策略),前者主要是对key value都是字符串的数据进行操作,后者的value可以放对象。本质上StringRedisTemplate继承的是RedisTemplate<String,String>。
使用默认的序列化策略JdkSerializationRedisSerializer操作集合对象List。
@SpringBootTest(classes = RedisstuApplication.class)
class Test01 {
@Autowired
private RedisTemplate redisTemplate;
@Test
void testRedisTemplate(){
ListOperations<String,User> listOps = redisTemplate.opsForList();
User user = new User("李四",18,new Date());
ArrayList<User> userList = new ArrayList<>();
userList.add(user);
userList.add(user);
listOps.leftPushAll("users",userList);
List<User> lists = listOps.range("users", 0, -1);
lists.forEach(x-> System.out.println(x.getUserName()+"=="+x.getBir()));
}
}
控制台看出可以正常获取对象。但是可视化工具里面可以看出实际存储的是JDK序列化后的数据。
使用Jackson2JsonRedisSerializer序列化规则操作集合对象。
- 修改RedisTemplate默认的序列化配置。修改后key使用StringRedisSerializer序列化规则,value使用Jackson2JsonRedisSerializer序列化规则。
package com.pec.redisstu.config;
import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.StringRedisSerializer;
import org.springframework.http.converter.json.Jackson2ObjectMapperBuilder;
@Configuration
public class RedisSerializableConfig {
@Bean
public RedisTemplate<String,Object> redisTemplate(LettuceConnectionFactory lettuceConnectionFactory){
//默认的序列化策略是JDK的,所以在存储的时候键和值都会先被序列化后再存储
RedisTemplate<String,Object> template = new RedisTemplate<>();
template.setConnectionFactory(lettuceConnectionFactory);
//1.修改键的序列化策略
template.setKeySerializer(new StringRedisSerializer());
//2.修改hash类型的value中key的序列化方式
template.setHashKeySerializer(new StringRedisSerializer());
//4.设置value的序列化策略
Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<>(Object.class);
template.setValueSerializer(jackson2JsonRedisSerializer);
//5.修改hash类型value中value的序列化方式
template.setHashValueSerializer(jackson2JsonRedisSerializer);
ObjectMapper objectMapper = Jackson2ObjectMapperBuilder.json().build();
objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
template.afterPropertiesSet();
return template;
}
}
测试时发现,当获取key对应的值,集合类型是List<Object>,无法转换成List<User>。
修改类型,改为泛型约束
java.lang.ClassCastException: java.util.LinkedHashMap cannot be cast to com.pec.redisstu.entity.User
at java.util.ArrayList.forEach(ArrayList.java:1249)
at com.pec.redisstu.Test01.testRedisTemplate(Test01.java:32)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.junit.platform.commons.util.ReflectionUtils.invokeMethod(ReflectionUtils.java:675)
at org.junit.jupiter.engine.execution.MethodInvocation.proceed(MethodInvocation.java:60)
at org.junit.jupiter.engine.execution.InvocationInterceptorChain$ValidatingInvocation.proceed(InvocationInterceptorChain.java:125)
at org.junit.jupiter.engine.extension.TimeoutExtension.intercept(TimeoutExtension.java:132)
at org.junit.jupiter.engine.extension.TimeoutExtension.interceptTestableMethod(TimeoutExtension.java:124)
at org.junit.jupiter.engine.extension.TimeoutExtension.interceptTestMethod(TimeoutExtension.java:74)
at org.junit.jupiter.engine.execution.ExecutableInvoker$ReflectiveInterceptorCall.lambda$ofVoidMethod$0(ExecutableInvoker.java:115)
at org.junit.jupiter.engine.execution.ExecutableInvoker.lambda$invoke$0(ExecutableInvoker.java:105)
at org.junit.jupiter.engine.execution.InvocationInterceptorChain$InterceptedInvocation.proceed(InvocationInterceptorChain.java:104)
at org.junit.jupiter.engine.execution.InvocationInterceptorChain.proceed(InvocationInterceptorChain.java:62)
at org.junit.jupiter.engine.execution.InvocationInterceptorChain.chainAndInvoke(InvocationInterceptorChain.java:43)
at org.junit.jupiter.engine.execution.InvocationInterceptorChain.invoke(InvocationInterceptorChain.java:35)
at org.junit.jupiter.engine.execution.ExecutableInvoker.invoke(ExecutableInvoker.java:104)
at org.junit.jupiter.engine.execution.ExecutableInvoker.invoke(ExecutableInvoker.java:98)
at org.junit.jupiter.engine.descriptor.TestMethodTestDescriptor.lambda$invokeTestMethod$6(TestMethodTestDescriptor.java:202)
at org.junit.platform.engine.support.hierarchical.ThrowableCollector.execute(ThrowableCollector.java:73)
at org.junit.jupiter.engine.descriptor.TestMethodTestDescriptor.invokeTestMethod(TestMethodTestDescriptor.java:198)
at org.junit.jupiter.engine.descriptor.TestMethodTestDescriptor.execute(TestMethodTestDescriptor.java:135)
at org.junit.jupiter.engine.descriptor.TestMethodTestDescriptor.execute(TestMethodTestDescriptor.java:69)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.lambda$executeRecursively$5(NodeTestTask.java:135)
at org.junit.platform.engine.support.hierarchical.ThrowableCollector.execute(ThrowableCollector.java:73)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.lambda$executeRecursively$7(NodeTestTask.java:125)
at org.junit.platform.engine.support.hierarchical.Node.around(Node.java:135)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.lambda$executeRecursively$8(NodeTestTask.java:123)
at org.junit.platform.engine.support.hierarchical.ThrowableCollector.execute(ThrowableCollector.java:73)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.executeRecursively(NodeTestTask.java:122)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.execute(NodeTestTask.java:80)
at java.util.ArrayList.forEach(ArrayList.java:1249)
at org.junit.platform.engine.support.hierarchical.SameThreadHierarchicalTestExecutorService.invokeAll(SameThreadHierarchicalTestExecutorService.java:38)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.lambda$executeRecursively$5(NodeTestTask.java:139)
at org.junit.platform.engine.support.hierarchical.ThrowableCollector.execute(ThrowableCollector.java:73)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.lambda$executeRecursively$7(NodeTestTask.java:125)
at org.junit.platform.engine.support.hierarchical.Node.around(Node.java:135)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.lambda$executeRecursively$8(NodeTestTask.java:123)
at org.junit.platform.engine.support.hierarchical.ThrowableCollector.execute(ThrowableCollector.java:73)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.executeRecursively(NodeTestTask.java:122)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.execute(NodeTestTask.java:80)
at java.util.ArrayList.forEach(ArrayList.java:1249)
at org.junit.platform.engine.support.hierarchical.SameThreadHierarchicalTestExecutorService.invokeAll(SameThreadHierarchicalTestExecutorService.java:38)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.lambda$executeRecursively$5(NodeTestTask.java:139)
at org.junit.platform.engine.support.hierarchical.ThrowableCollector.execute(ThrowableCollector.java:73)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.lambda$executeRecursively$7(NodeTestTask.java:125)
at org.junit.platform.engine.support.hierarchical.Node.around(Node.java:135)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.lambda$executeRecursively$8(NodeTestTask.java:123)
at org.junit.platform.engine.support.hierarchical.ThrowableCollector.execute(ThrowableCollector.java:73)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.executeRecursively(NodeTestTask.java:122)
at org.junit.platform.engine.support.hierarchical.NodeTestTask.execute(NodeTestTask.java:80)
at org.junit.platform.engine.support.hierarchical.SameThreadHierarchicalTestExecutorService.submit(SameThreadHierarchicalTestExecutorService.java:32)
at org.junit.platform.engine.support.hierarchical.HierarchicalTestExecutor.execute(HierarchicalTestExecutor.java:57)
at org.junit.platform.engine.support.hierarchical.HierarchicalTestEngine.execute(HierarchicalTestEngine.java:51)
at org.junit.platform.launcher.core.DefaultLauncher.execute(DefaultLauncher.java:229)
at org.junit.platform.launcher.core.DefaultLauncher.lambda$execute$6(DefaultLauncher.java:197)
at org.junit.platform.launcher.core.DefaultLauncher.withInterceptedStreams(DefaultLauncher.java:211)
at org.junit.platform.launcher.core.DefaultLauncher.execute(DefaultLauncher.java:191)
at org.junit.platform.launcher.core.DefaultLauncher.execute(DefaultLauncher.java:128)
at com.intellij.junit5.JUnit5IdeaTestRunner.startRunnerWithArgs(JUnit5IdeaTestRunner.java:74)
at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)
打印list得到的数据:
[{userName=李四, age=18, bir=1608009991899}, {userName=李四, age=18, bir=1608009991899}]
- 可以发现这个lists里面并不是user对象,我们前面在配置的时候配置value的序列化规则
template.setValueSerializer(jackson2JsonRedisSerializer);
所以这里要把json串转成bean对象的格式。- 使用jackson objectMapper中的convertValue()方法转换。
objectMapper.convertValue(lists, new TypeReference<List<User>>() { });
package com.pec.redisstu;
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.pec.redisstu.entity.User;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.data.redis.core.ListOperations;
import org.springframework.data.redis.core.RedisTemplate;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
@SpringBootTest(classes = RedisstuApplication.class)
class Test01 {
@Autowired
private RedisTemplate<String,? extends Object> redisTemplate;
@Autowired
ObjectMapper objectMapper;
@Test
void testRedisTemplate(){
ListOperations<String,User> listOps = (ListOperations<String, User>) redisTemplate.opsForList();
User user = new User("李四",18,new Date());
ArrayList<User> userList = new ArrayList<>();
userList.add(user);
userList.add(user);
listOps.leftPushAll("users",userList);
//此时拿到的是jackson序列化后的json字符串
List<User> lists = listOps.range("users", 0, -1);
//jackson解析出具体的bean
List<User> users= objectMapper.convertValue(lists, new TypeReference<List<User>>() { });
users.forEach(x-> System.out.println(x.getUserName()+"=="+x.getBir()));
}
}
打开可视化工具,在里面就可以看到具体的key和值了。
如有错误,请指正!!!
本文参考:
https://www.cnblogs.com/lossingdawn/p/6722854.html
https://blog.csdn.net/weixin_39778417/article/details/103304940