Redis的springboot整合

1,785 阅读3分钟

本文正在参加「金石计划 . 瓜分6万现金大奖」

Redis的springboot整合

1、springboot整合

  1. 导入依赖
 <dependency>
     <groupId>org.springframework.boot</groupId>
     <artifactId>spring-boot-starter-data-redis</artifactId>
 </dependency>
 ​

springboot 2.x后 ,原来使用的 Jedis 被 lettuce 替换。

jedis:采用的直连,多个线程操作的话,是不安全的。如果要避免不安全,使用jedis pool连接池!更像BIO模式

lettuce:采用netty,实例可以在多个线程中共享,不存在线程不安全的情况!可以减少线程数据了,更像NIO模式

我们在学习SpringBoot自动配置的原理时,整合一个组件并进行配置一定会有一个自动配置类xxxAutoConfiguration,并且在spring.factories中也一定能找到这个类的完全限定名。Redis也不例外。 image-20221120152029750

那么就一定还存在一个RedisProperties类

image-20221120152104787

之前我们说SpringBoot2.x后默认使用Lettuce来替换Jedis,现在我们就能来验证了。

先看Jedis:

image-20221120152122770

@ConditionalOnClass注解中有两个类是默认不存在的,所以Jedis是无法生效的

然后再看Lettuce:

image-20221120152147575

完美生效。

现在我们回到RedisAutoConfiguratio

image-20221120152216326

只有两个简单的Bean

  • RedisTemplate
  • StringRedisTemplate

当看到xxTemplate时可以对比RestTemplat、SqlSessionTemplate,通过使用这些Template来间接操作组件。那么这俩也不会例外。分别用于操作Redis和Redis中的String数据类型。

在RedisTemplate上也有一个条件注解,说明我们是可以对其进行定制化的

说完这些,我们需要知道如何编写配置文件然后连接Redis,就需要阅读RedisProperties

image-20221120152249242

这是一些基本的配置属性。

image-20221120152305340

还有一些连接池相关的配置。注意使用时一定使用Lettuce的连接池。

image-20221120152330995

  1. 编写配置文件
 # 配置redis
 spring.redis.host=39.99.xxx.xx
 spring.redis.port=6379
 ​
  1. 使用RedisTemplate
 @SpringBootTest
 class Redis02SpringbootApplicationTests {
 ​
     @Autowired
     private RedisTemplate redisTemplate;
 ​
     @Test
     void contextLoads() {
 ​
         // redisTemplate 操作不同的数据类型,api和我们的指令是一样的
         // opsForValue 操作字符串 类似String
         // opsForList 操作List 类似List
         // opsForHah
 ​
         // 除了基本的操作,我们常用的方法都可以直接通过redisTemplate操作,比如事务和基本的CRUD
 ​
         // 获取连接对象
         //RedisConnection connection = redisTemplate.getConnectionFactory().getConnection();
         //connection.flushDb();
         //connection.flushAll();
 ​
         redisTemplate.opsForValue().set("mykey","kuangshen");
         System.out.println(redisTemplate.opsForValue().get("mykey"));
     }
 }
 ​
  1. 测试结果

此时我们回到Redis查看数据时候,惊奇发现全是乱码,可是程序中可以正常输出:

image-20221120152435786

这时候就关系到存储对象的序列化问题,在网络中传输的对象也是一样需要序列化,否者就全是乱码。

我们转到看那个默认的RedisTemplate内部什么样子:

image-20221120152449453

在最开始就能看到几个关于序列化的参数。

默认的序列化器是采用JDK序列化器

image-20221120152510241

而默认的RedisTemplate中的所有序列化器都是使用这个序列化器:

image-20221120152525109

后续我们定制RedisTemplate就可以对其进行修改。

RedisSerializer提供了多种序列化方案:

  • 直接调用RedisSerializer的静态方法来返回序列化器,然后set

image-20221120152542599

  • 自己new 相应的实现类,然后set

image-20221120152604500

  1. 定制RedisTemplate的模板:

我们创建一个Bean加入容器,就会触发RedisTemplate上的条件注解使默认的RedisTemplate失效。

 @Configuration
 public class RedisConfig {
 ​
    @Bean
     public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) throws UnknownHostException {
         // 将template 泛型设置为 <String, Object>
         RedisTemplate<String, Object> template = new RedisTemplate();
         // 连接工厂,不必修改
         template.setConnectionFactory(redisConnectionFactory);
         /*
          * 序列化设置
          */
         // key、hash的key 采用 String序列化方式
         template.setKeySerializer(RedisSerializer.string());
         template.setHashKeySerializer(RedisSerializer.string());
         // value、hash的value 采用 Jackson 序列化方式
         template.setValueSerializer(RedisSerializer.json());
         template.setHashValueSerializer(RedisSerializer.json());
         template.afterPropertiesSet();
         
         return template;
     }
 }
 ​

这样一来,只要实体类进行了序列化,我们存什么都不会有乱码的担忧了。

自定义Redis工具类

使用RedisTemplate需要频繁调用.opForxxx然后才能进行对应的操作,这样使用起来代码效率低下,工作中一般不会这样使用,而是将这些常用的公共API抽取出来封装成为一个工具类,然后直接使用工具类来间接操作Redis,不但效率高并且易用。

工具类参考博客:

www.cnblogs.com/zeng1994/p/…

www.cnblogs.com/zhzhlong/p/…