SpringBoot集成Redis的實(shí)現(xiàn)示例
前面一篇文章已經(jīng)寫了如何搭建一個(gè)單機(jī)版Redis服務(wù), 那么我們應(yīng)該怎么在現(xiàn)有的系統(tǒng)中集成進(jìn)來(lái)呢? 由于筆者使用的編程語(yǔ)言是Java, 所以本篇文章主要描述SpringBoot如何集成單Redis節(jié)點(diǎn)完成數(shù)據(jù)的增刪改查.
SpringBoot環(huán)境
快速搭建一個(gè)SpringBoot工程
進(jìn)入 https://start.spring.io
網(wǎng)站, 使用該網(wǎng)站初始化一個(gè)SpringBoot工程
添加相關(guān)依賴
因?yàn)槭褂胹pring initializer已經(jīng)幫我們把Redis的依賴建立好了; 但是由于我們要使用Jedis客戶端訪問(wèn)Redis, 所以還需要添加Jedis的依賴;
<dependency> <groupId>redis.clients</groupId> <artifactId>jedis</artifactId> <version>2.9.0</version> //版本號(hào)可以放在properties中作為屬性, 這邊用${jedis.version}來(lái)依賴 </dependency>
配置Redis節(jié)點(diǎn)信息
打開(kāi)application.properties文件, 初始化的文件是空的; 我們將spring redis最基本的信息加入進(jìn)去
spring.redis.host=localhost spring.redis.port=6379
將Redis信息讀入到程序中
新建一個(gè)Java類命名為StandaloneRedisConfig.java
, 放在com.xxx.example.config
包下
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"})
如果只是單機(jī)的Redis則不需要添加該屬性; 但是為了后面一套代碼兼容多個(gè)Redis部署模式, 使用該屬性作為是否創(chuàng)建Bean的條件; 如果是集群模式那么就不會(huì)使用spring.redis.host
來(lái)作為連接字符串了;
配置Jedis的連接池
將Redis連接對(duì)象放入到Spring容器中進(jìn)行管理
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
是一個(gè)道理
這里的scanBasePackages = {"com.terrylmay.redis.example"}
是為了以后將Redis的客戶端獨(dú)立出一個(gè)工程而做的, 當(dāng)然獨(dú)立出來(lái)的工程base包名還要是這個(gè)才可以;
因?yàn)檫€沒(méi)有看Redis支持的數(shù)據(jù)結(jié)構(gòu), 那么現(xiàn)在只是把Redis字符串模板類放到Spring 容器中, 后續(xù)再增加其他數(shù)據(jù)類型的支持;
創(chuàng)建操作Redis的接口 以及實(shí)現(xiàn)
創(chuàng)建ICacheProvider.java
接口:
package com.terrylmay.redis.example.provider; public interface ICacheProvider { void setString(String key, String value); String getString(String key); }
Jedis版本的實(shí)現(xiàn):
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); } }
這樣基本上一個(gè)可以操作Redis的Java程序就已經(jīng)就緒了; 那么我們需要驗(yàn)證一下, 當(dāng)然如果在主工程中寫一個(gè)類去驗(yàn)證也是沒(méi)有問(wèn)題的, 比如創(chuàng)建一個(gè)Bean, 并且放到被PostContruct
注解的方法里面;
但是更加專業(yè)的做法是寫一個(gè)測(cè)試程序來(lái)測(cè)試, 下面看一下該測(cè)試程序應(yīng)該怎么寫
UT測(cè)試程序可用性
因?yàn)閯?chuàng)建工程的時(shí)候, 就已經(jīng)有一個(gè)測(cè)試類在test目錄下面了, 我們?cè)黾游覀兿胍墓δ?/p>
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或者直接斷言來(lái)處理
(本來(lái)想用markdown語(yǔ)法來(lái)標(biāo)紅的, 但是發(fā)現(xiàn)簡(jiǎn)書(shū)竟然不支持html的寫法; 沒(méi)辦法只能用``來(lái)搞定了)
開(kāi)發(fā)過(guò)程中遇到的問(wèn)題
一、在寫好所有的程序之后, 跑測(cè)試用例, 但是始終都是報(bào)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)}
原因共有三點(diǎn):
1、寫了scanBasepackages
來(lái)掃描包下面的bean, 掃描的包與類所在的包不一樣, 只有一個(gè)字符之差 com.terrylmay.redis.example
與 com.terrlmay.redis.example
, 當(dāng)然這時(shí)候idea
會(huì)報(bào)錯(cuò), 只是我不認(rèn)識(shí)那個(gè)錯(cuò)而已; Idea報(bào)錯(cuò)如圖所示:
2、按照網(wǎng)上的application.properties屬性的讀取方式, 只使用了一個(gè)注解:
@ConfigurationProperties(prefix = "spring.redis")
但是點(diǎn)進(jìn)該注解里面看, 它其實(shí)并沒(méi)有Component注解的功能; 所以增加了@Configuration
注解
3、第三個(gè)原因不仔細(xì)斷然不會(huì)發(fā)現(xiàn)這個(gè)錯(cuò)誤
我理解的是只要工程里面父工程是spring-boot-starter-parent
, 那么就不應(yīng)該存在這類Jar包沒(méi)有依賴的問(wèn)題, 打開(kāi)文檔
依賴可粘貼版:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-configuration-processor</artifactId> <optional>true</optional> </dependency>
到這里, 一個(gè)能夠使用Redis的Java工程就已經(jīng)就緒了; 最終的代碼全部在Github的spring-redis-example倉(cāng)庫(kù)下, 歡迎star與PR
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Springboot2.X集成redis集群(Lettuce)連接的方法
- SpringBoot集成Redisson實(shí)現(xiàn)分布式鎖的方法示例
- SpringBoot利用redis集成消息隊(duì)列的方法
- 淺談SpringBoot集成Redis實(shí)現(xiàn)緩存處理(Spring AOP實(shí)現(xiàn))
- SpringBoot集成Redis實(shí)現(xiàn)消息隊(duì)列的方法
- 使用SpringBoot集成redis的方法
- SpringBoot集成Redisson實(shí)現(xiàn)延遲隊(duì)列的場(chǎng)景分析
- SpringBoot集成redis的示例代碼
相關(guān)文章
解決Netty解碼http請(qǐng)求獲取URL亂碼問(wèn)題
這篇文章主要介紹了解決Netty解碼http請(qǐng)求獲取URL亂碼問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-06-06Java讀文件修改默認(rèn)換行符的實(shí)現(xiàn)
這篇文章主要介紹了Java讀文件修改默認(rèn)換行符的實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12java的main方法中調(diào)用spring的service方式
這篇文章主要介紹了在java的main方法中調(diào)用spring的service方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12Spring攔截器HandlerInterceptor接口代碼解析
這篇文章主要介紹了Spring攔截器HandlerInterceptor接口代碼解析,具有一定借鑒價(jià)值,需要的朋友可以參考下2017-12-12