深刻理解Redis系列之SpringBoot集成Redis

前面一篇文章已經寫了如何搭建一個單機版Redis服務, 那麼咱們應該怎麼在現有的系統中集成進來呢? 因爲筆者使用的編程語言是Java, 因此本篇文章主要描述SpringBoot如何集成單Redis節點完成數據的增刪改查.

SpringBoot環境

快速搭建一個SpringBoot工程

進入 https://start.spring.io 網站, 使用該網站初始化一個SpringBoot工程html

1706159233-5bdc6766067a1.png

添加相關依賴

由於使用spring initializer已經幫咱們把Redis的依賴創建好了; 可是因爲咱們要使用Jedis客戶端訪問Redis, 因此還須要添加Jedis的依賴;java

<dependency>
            <groupId>redis.clients</groupId>
            <artifactId>jedis</artifactId>
            <version>2.9.0</version> //版本號能夠放在properties中做爲屬性, 這邊用${jedis.version}來依賴
     </dependency>

配置Redis節點信息

打開application.properties文件, 初始化的文件是空的; 咱們將spring redis最基本的信息加入進去git

spring.redis.host=localhost
spring.redis.port=6379

將Redis信息讀入到程序中

新建一個Java類命名爲StandaloneRedisConfig.java, 放在com.xxx.example.config包下github

package com.terrylmay.redis.example.config;

import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Configuration;

@Configuration
@ConfigurationProperties(prefix = "spring.redis")
@ConditionalOnProperty(name = {"spring.redis.host"})
public class StandaloneRedisConfig {

    String host;

    int port;

    public String getHost() {
        return host;
    }

    public void setHost(String host) {
        this.host = host;
    }

    public int getPort() {
        return port;
    }

    public void setPort(int port) {
        this.port = port;
    }
}

上面配置中的@ConditionalOnProperty(name = {"spring.redis.host"}) 若是隻是單機的Redis則不須要添加該屬性; 可是爲了後面一套代碼兼容多個Redis部署模式, 使用該屬性做爲是否建立Bean的條件; 若是是集羣模式那麼就不會使用spring.redis.host來做爲鏈接字符串了;redis

配置Jedis的鏈接池

將Redis鏈接對象放入到Spring容器中進行管理spring

package com.terrylmay.redis.example;

import com.terrylmay.redis.example.config.StandaloneRedisConfig;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
import org.springframework.context.annotation.Bean;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.connection.RedisStandaloneConfiguration;
import org.springframework.data.redis.connection.jedis.JedisConnectionFactory;
import org.springframework.data.redis.core.StringRedisTemplate;

@SpringBootApplication(scanBasePackages = {"com.terrylmay.redis.example"})
public class RedisExampleApplication {

    public static void main(String[] args) {
        SpringApplication.run(RedisExampleApplication.class, args);
    }

    @Autowired
    StandaloneRedisConfig standaloneRedisConfig;

    @Autowired
    RedisConnectionFactory redisConnectionFactory;

    @Bean
    @ConditionalOnBean(value = {StandaloneRedisConfig.class})
    public RedisConnectionFactory standaloneRedisConnectionFactory() {
        JedisConnectionFactory factory = new JedisConnectionFactory(new RedisStandaloneConfiguration(standaloneRedisConfig.getHost(), standaloneRedisConfig.getPort()));
        return factory;
    }

    @Bean
    public StringRedisTemplate stringRedisTemplate() {
        return new StringRedisTemplate(redisConnectionFactory);
    }
}

這裏的@ConditionalOnBean(value = {StandaloneRedisConfig.class})與上面的ConditionalOnProperty 是一個道理編程

這裏的scanBasePackages = {"com.terrylmay.redis.example"} 是爲了之後將Redis的客戶端獨立出一個工程而作的, 固然獨立出來的工程base包名還要是這個才能夠;markdown

由於尚未看Redis支持的數據結構, 那麼如今只是把Redis字符串模板類放到Spring 容器中, 後續再增長其餘數據類型的支持;數據結構

建立操做Redis的接口 以及實現

建立ICacheProvider.java接口:app

package com.terrylmay.redis.example.provider;

public interface ICacheProvider {
    void setString(String key, String value);

    String getString(String key);
}

Jedis版本的實現:

package com.terrylmay.redis.example.provider.impl;

import com.terrylmay.redis.example.provider.ICacheProvider;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Component;

@Component
public class JedisCacheProvider implements ICacheProvider {

    @Autowired
    StringRedisTemplate stringRedisTemplate;

    @Override
    public void setString(String key, String value) {
        stringRedisTemplate.opsForValue().set(key, value);
    }

    @Override
    public String getString(String key) {
        return stringRedisTemplate.opsForValue().get(key);
    }
}

這樣基本上一個能夠操做Redis的Java程序就已經就緒了; 那麼咱們須要驗證一下, 固然若是在主工程中寫一個類去驗證也是沒有問題的, 好比建立一個Bean, 而且放到被PostContruct註解的方法裏面;

可是更加專業的作法是寫一個測試程序來測試, 下面看一下該測試程序應該怎麼寫

UT測試程序可用性

由於建立工程的時候, 就已經有一個測試類在test目錄下面了, 咱們增長咱們想要的功能

package com.terrylmay.redis.example;

import com.terrylmay.redis.example.provider.ICacheProvider;
import org.junit.Assert;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

@RunWith(SpringJUnit4ClassRunner.class)
@SpringBootTest(classes = {RedisExampleApplication.class})
public class RedisExampleApplicationTests {

    @Autowired
    ICacheProvider jedisCacheProvider;

    @Test
    public void contextLoads() {
        jedisCacheProvider.setString("name", "terrylmay");
        System.out.println(jedisCacheProvider.getString("name")); 
        Assert.assertEquals("terrylmay", jedisCacheProvider.getString("name"));
    }
}

注: 程序中不要有打印, 使用Logger或者直接斷言來處理 (原本想用markdown語法來標紅的, 可是發現簡書居然不支持html的寫法; 沒辦法只能用``來搞定了)

開發過程當中遇到的問題

1、在寫好全部的程序以後, 跑測試用例, 可是始終都是報NoSuchBeanException

Caused by: org.springframework.beans.factory.NoSuchBeanDefinitionException:
No qualifying bean of type 'com.terrylmay.redis.example.config.StandaloneRedisConfig' available: expected at least 1 bean which qualifies as autowire candidate. 
Dependency annotations: {
@org.springframework.beans.factory.annotation.Autowired(required=true)}

緣由共有三點:

一、寫了scanBasepackages來掃描包下面的bean, 掃描的包與類所在的包不同, 只有一個字符之差 com.terrylmay.redis.examplecom.terrlmay.redis.example, 固然這時候idea會報錯, 只是我不認識那個錯而已; Idea報錯如圖所示:

image.png

二、按照網上的application.properties屬性的讀取方式, 只使用了一個註解:
@ConfigurationProperties(prefix = "spring.redis") 可是點進該註解裏面看, 它其實並無Component註解的功能; 因此增長了@Configuration註解

三、第三個緣由不仔細斷然不會發現這個錯誤

image.png

我理解的是隻要工程裏面父工程是spring-boot-starter-parent, 那麼就不該該存在這類Jar包沒有依賴的問題, 打開文檔

image.png

依賴可粘貼版:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-configuration-processor</artifactId>
    <optional>true</optional>
</dependency>

到這裏, 一個可以使用Redis的Java工程就已經就緒了; 最終的代碼所有在Github的spring-redis-example倉庫下, 歡迎star與PR

相關文章
相關標籤/搜索