🎨RedisTemplate泛型使用

RedisTemplate<String,String>与StringRedisTemplate底层对象是同一个

java redis 泛型 redistemplate泛型封装_java

RedisTemplate<Object,Object>泛型不同,对象不同

java redis 泛型 redistemplate泛型封装_spring boot_02

既然是泛型就可以指定类型,通常K为String,V为对象或其他类型

User类(可序列化)

@Data@AllArgsConstructor @NoArgsConstructor @Builder public class User implements Serializable { private String name; private int age; }

@Autowired 与 @Resource 区别

  • @Autowired注解由Spring提供,只按照byType注入;
  • @Resource注解由J2EE提供,默认按照byName自动注入,按照名字如果找不到,再按照类型byType去找
  • @Autowired默认按类型进行装配,
  • @Resource默认按照名称进行装配。
//@Autowired 按类型找不到会报错,找不到bean//@Resource 按名称redisTemplate找,若找不到按照类型也找不到则报错
@Resource(name = "redisTemplate")//不设置name默认安装方法名找
private RedisTemplate<String, User> redisTemplate;
因此使用@Resource(name = "redisTemplate")来注入bean
@Resource(name = "redisTemplate")    private RedisTemplate<String, User> redisTemplate;

    @Test
    public void test1(){
        User user1 = User.builder().name("张三").age(23).build();
        //会出现乱码问题
        redisTemplate.opsForValue().set("user1", user1);
        //可以正常获取
        User user = redisTemplate.opsForValue().get("user1");
        System.out.println(user);
    }

java redis 泛型 redistemplate泛型封装_redis_03

乱码问题分析

StringRedisTemplate不会乱码是因为底层使用的是StringRedisSerializer序列化工具

java redis 泛型 redistemplate泛型封装_redis_04

RedisTemplate<Object,Object>乱码原因是没有使用序列化工具,底层默认使用原生的jdk序列化工具JdkSerializationRedisSerializer从而乱码

java redis 泛型 redistemplate泛型封装_序列化_05

具体原因就是出在JdkSerializationRedisSerializer上

java redis 泛型 redistemplate泛型封装_序列化_06

解决乱码问题

自定义RedisConfig

@Configurationpublic class RedisConfig {
    @Bean
    public RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) {
        RedisTemplate<Object, Object> template = new RedisTemplate<>();
        template.setConnectionFactory(redisConnectionFactory);
        //配置序列化工具
        //json序列化工具 对象使用此序列化
        Jackson2JsonRedisSerializer jsonSerializer = new Jackson2JsonRedisSerializer(Object.class);
        //类型转换
        ObjectMapper objectMapper = new ObjectMapper();
        //设置所有属性可见性序列化
        objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
        //默认键入全包名
        objectMapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
        jsonSerializer.setObjectMapper(objectMapper);
        //string序列化工具 string使用此序列化
        StringRedisSerializer stringSerializer = new StringRedisSerializer();
        //设置KEY/key的序列化工具为stringSerializer
        template.setKeySerializer(stringSerializer);
        //设置值的序列化工具为jsonSerializer
        template.setValueSerializer(jsonSerializer);
        //设置hash的小key的序列化工具为stringSerializer
        template.setHashKeySerializer(stringSerializer);
        //设置hash的值的序列化工具为jsonSerializer
        template.setHashValueSerializer(jsonSerializer);
        template.afterPropertiesSet();
        return template;
    }
}

对象使用json序列化工具Jackson2JsonRedisSerializer

String还是使用StringRedisSerializer序列化工具

之所以我们自定义的RedisTemplate<Object, Object>类型的bean会覆盖官方底层源码的bean,是因为官方设置了当没有注册redistemplate这个bean时,才使用此官方的bean

@ConditionalOnMissingBean( name = {"redisTemplate"})

java redis 泛型 redistemplate泛型封装_java redis 泛型_07

 解决乱码问题后再执行test1

java redis 泛型 redistemplate泛型封装_java_08