SpringCloud Ribbon與OpenFeign詳解如何實(shí)現(xiàn)服務(wù)調(diào)用
Ribbon
初識Ribbon
Ribbon是什么
Ribbon是Netflix發(fā)布的開源項(xiàng)目,主要功能是提供對客戶端進(jìn)行負(fù)載均衡算法的一套工具,將Netflix的中間層服務(wù)連接在一起。Ribbon客戶端組件提供一系列完善的配置項(xiàng)如連接超時(shí),重試等。簡單的說,就是在配置文件中列出Load Balancer(簡稱LB)后面所有的機(jī)器,Ribbon會自動的幫助你基于某種規(guī)則(如簡單輪詢,隨即連接等)去連接這些機(jī)器。我們也可以使用Ribbon實(shí)現(xiàn)自定義的負(fù)載均衡算法。
Ribbon能干什么
前面提到說Ribbon的作用就是向客戶端提供負(fù)載均衡算法的工具,那么什么是負(fù)載均衡呢?負(fù)載均衡就是將用戶發(fā)來的請求通過算法均攤到多個(gè)服務(wù)上,從而達(dá)到系統(tǒng)的HA(高可用性)
其中,負(fù)載均衡又可分為本地負(fù)載均衡(進(jìn)程內(nèi)LB)和服務(wù)端負(fù)載均衡(集中式LB),服務(wù)端負(fù)載均衡以Nginx為例,用戶的所有請求都會交給Nginx,由其決定請求將被轉(zhuǎn)發(fā)到哪個(gè)服務(wù)器;Ribbon是本地負(fù)載均衡,在調(diào)用接口的時(shí)候從 eureka 注冊中心服務(wù)器端上獲取服務(wù)注冊信息列表緩存到本地,從而可以在本地實(shí)現(xiàn)RPC遠(yuǎn)程調(diào)用服務(wù)
使用Ribbon實(shí)現(xiàn)負(fù)載均衡
實(shí)際上Ribbon可以簡單的理解為負(fù)載均衡算法 + RestTemplate的調(diào)用,也就是說想要使用Ribbon實(shí)現(xiàn)負(fù)載均衡,就可以通過這兩個(gè)技術(shù)加以實(shí)現(xiàn)。
RestTemplate三步走
第一步: 引入Ribbon場景啟動器依賴,但是之前使用eureka的時(shí)候我們在pom文件中導(dǎo)入過netflix-eureka-server的依賴,其中就默認(rèn)引入了ribbon的場景啟動器依賴(netflix-eureka-client也會默認(rèn)導(dǎo)入),如果再引一次也可以,但是真沒那必要。
第二步: 使用配置文件開啟 RestTemplate
/** * @ClassName: ApplicationContextConfig * @Description: 程序配置類,用于開啟RestTemplate服務(wù),以供后面使用 * @author: chenhao * @date: 2022/7/17 */ @Configuration public class ApplicationContextConfig { @Bean @LoadBalanced public RestTemplate getRestTemplate() { return new RestTemplate(); } }
第三步: 使用RestTemplate的API實(shí)現(xiàn)負(fù)載均衡,RestTemplate的API根據(jù)請求方式的不同可以被分為get和post,根據(jù)返回類型又可以分為Object(響應(yīng)體轉(zhuǎn)化成的json串)和Entity(響應(yīng)的重要信息,包括響應(yīng)頭、狀態(tài)碼、響應(yīng)體等,可以使用對應(yīng)的get方法獲取到值),所以說最主要的四個(gè)API是getForObject、postForObject、getForEntity、postForEntity,這里我把四種API的使用方法都向大家介紹一下
@RestController @Slf4j @RequestMapping("consumer") @Api("消費(fèi)者的訂單管理類") public class OrderController { // 先注入RestTemplate對象 @Resource private RestTemplate restTemplate; // 使用API實(shí)現(xiàn)負(fù)載均衡 @ApiOperation(value = "創(chuàng)建一條支付記錄", tags = ApiVersionConstant.v1_0) @PostMapping(value = "/payment/create", produces = {"application/json;charset=UTF-8"}) public CommonResult<Integer> create(@RequestBody Payment payment) { return restTemplate.postForObject(UrlConstant.CLUSTER_PAYMENT_URL + "/payment/create", payment, CommonResult.class); } @ApiOperation(value = "根據(jù)ID查詢支付記錄", tags = ApiVersionConstant.v1_0) @GetMapping(value = "/payment/get/{id}", produces = {"application/json;charset=UTF-8"}) public CommonResult<Payment> getPayment(@PathVariable("id") Integer id) { return restTemplate.getForObject(UrlConstant.CLUSTER_PAYMENT_URL + "/payment/get/" + id, CommonResult.class); } @ApiOperation(value = "根據(jù)ID查詢支付記錄Entity", tags = ApiVersionConstant.v1_0) @GetMapping(value = "/payment/getForEntity/{id}", produces = {"application/json;charset=UTF-8"}) public CommonResult<Payment> getPaymentEntity(@PathVariable("id") Integer id) { ResponseEntity<CommonResult> entity = restTemplate.getForEntity(UrlConstant.CLUSTER_PAYMENT_URL + "/payment/get/" + id, CommonResult.class); if (entity.getStatusCode().is2xxSuccessful()) { return entity.getBody(); } else { return new CommonResult<>(444, "操作失敗"); } } @ApiOperation(value = "創(chuàng)建一條支付記錄Entity", tags = ApiVersionConstant.v1_0) @PostMapping(value = "/payment/createEntity", produces = {"application/json;charset=UTF-8"}) public CommonResult<Integer> createEntity(@RequestBody Payment payment) { return restTemplate.postForEntity(UrlConstant.CLUSTER_PAYMENT_URL + "/payment/create", payment, CommonResult.class) .getBody(); } }
負(fù)載均衡算法
經(jīng)過我們上面的嘗試,不難發(fā)現(xiàn)使用RestTemplate實(shí)現(xiàn)的負(fù)載均衡算法是輪詢機(jī)制,實(shí)際上IRule中不僅僅只提供了一種算法
IRule實(shí)現(xiàn)算法切換
第一步: 新建一個(gè)package,在官方文檔中聲明了IRule的配置類不能放到@ComponentScan注解所能掃描到的當(dāng)前包以及子包下,否則自定義的配置類就會被Ribbon的所有客戶端所共享,以至于失去客戶端定制化的可能性。主程序入口上的@SpringBootApplication注解是復(fù)合注解,其中就包含@ComponentScan注解,而且是直接掃描主程序入口所在的當(dāng)前包以及子包,也就是說配置類必須放到主程序入口之外的包下,于是需要新建一個(gè)package
第二步: 在新建的包中新建一個(gè)MySelfRule規(guī)則類,用于設(shè)置輪詢算法,如果不設(shè)置的話就默認(rèn)為輪詢
@Configuration public class MySelfRule { @Bean public IRule myRule() { // 修改輪詢算法為隨機(jī)算法 return new RandomRule(); } }
第三步: 主啟動類上使用@RibbonClient(name = “CLOUD-PAYMENT-SERVICE”, configuration = MySelfRule.class)注解,用于指定應(yīng)用服務(wù)和自定義算法規(guī)則的配置類
@SpringBootApplication @EnableEurekaClient @RibbonClient(name = "CLOUD-PAYMENT-SERVICE", configuration = MySelfRule.class) public class OrderMain80 { public static void main(String[] args) { SpringApplication.run(OrderMain80.class, args); } }
輪詢算法
所謂的輪詢算法就是根據(jù)所有的服務(wù),依次將請求均攤到所有的服務(wù)依次訪問,它的算法實(shí)現(xiàn)就是用RestTemplate接收到的請求數(shù)量對服務(wù)器集群的數(shù)量進(jìn)行取模運(yùn)算,余數(shù)就是服務(wù)在服務(wù)列表中對應(yīng)的索引位置,所以說可以實(shí)現(xiàn)輪詢。但是如果中途服務(wù)器關(guān)掉的話接收到的請求數(shù)量就會從1重新計(jì)數(shù)
OpenFeign
初識OpenFeign
什么是OpenFeign
openFeign是要聲明式的web服務(wù)客戶端,或叫做聲明式REST客戶端,它讓編寫web服務(wù)客戶端變得簡單。它將提供者的restful服務(wù)偽裝成接口進(jìn)行消費(fèi),消費(fèi)者只需要通過feign接口+注解就可以直接調(diào)用提供者的服務(wù)接口,也就是可以實(shí)現(xiàn)接口對接口的調(diào)用,而無需像ribbon一樣通過restTemplate方式對提供者的服務(wù)進(jìn)行調(diào)用
值得注意的一點(diǎn)是,openFeign內(nèi)置了負(fù)載均衡器-Ribbon,所以說openfeign也可以使用負(fù)載均衡算法
如何使用OpenFeign
第一步: 引入相關(guān)依賴
<!--openfeign--> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-openfeign</artifactId> </dependency>
第二步: 配置配置文件,只是基本配置和注冊,沒有OpenFeign獨(dú)有的配置
server:
port: 80
eureka:
client:
register-with-eureka: false
service-url:
#服務(wù)端的地址,服務(wù)端為集群版,向所有的模塊都注冊
defaultZone: http://localhost:7001/eureka,
http://localhost:7002/eureka,
http://localhost:7003/eureka
第三步: 主啟動類開啟OpenFeign客戶端
@SpringBootApplication @EnableFeignClients public class OrderFeignMain80 { public static void main(String[] args) { SpringApplication.run(OrderFeignMain80.class, args); } }
第四步: 之前使用ribbon是直接在controller里調(diào)用payment服務(wù)的controller,但是openfeign則是通過service調(diào)用,于是第三步就是創(chuàng)建一個(gè)service接口用于調(diào)用payment服務(wù)的接口
@Component @FeignClient(value = "CLOUD-PAYMENT-SERVICE") // 用于指定服務(wù)名,可在eureka或者服務(wù)的配置文件中查看 public interface PaymentFeignService { @GetMapping(value = "/payment/get/{id}", produces = {"application/json;charset=UTF-8"}) CommonResult<Payment> getPaymentById(@PathVariable("id") Integer id); }
第五步: controller層調(diào)用service接口
@RestController @RequestMapping("consumer") public class OrderFeignController { @Autowired private PaymentFeignService paymentFeignService; @GetMapping(value = "/payment/get/{id}", produces = {"application/json;charset=UTF-8"}) public CommonResult<Payment> getPaymentById(@PathVariable("id") Integer id) { return paymentFeignService.getPaymentById(id); } }
如此操作也可實(shí)現(xiàn)order服務(wù)對payment服務(wù)的調(diào)用,而且由于OpenFeign默認(rèn)引入Ribbon,去進(jìn)行order服務(wù)訪問的時(shí)候,后端會默認(rèn)輪詢名為是“CLOUD-PAYMENT-SERVICE”的微服務(wù),也就是兩個(gè)payment服務(wù)
OpenFeign超時(shí)控制
使用OpenFeign調(diào)用服務(wù)接口,默認(rèn)等待時(shí)間為1秒,超時(shí)就會直接報(bào)錯(cuò)。如果有些服務(wù)的調(diào)用確實(shí)會花費(fèi)超過1s的時(shí)間,就需要我們在服務(wù)調(diào)用方(也就是order服務(wù))的配置文件中進(jìn)行配置
由于OpenFeign的超時(shí)控制由其底層的ribbon實(shí)現(xiàn),于是配置文件中的超時(shí)控制也由ribbon進(jìn)行配置
#設(shè)置feign客戶端超時(shí)時(shí)間(OpenFeign默認(rèn)支持ribbon)
ribbon:
#指的是建立連接所用的時(shí)間,適用于網(wǎng)絡(luò)狀況正常的情況下, 兩端連接所用的時(shí)間
ReadTimeout: 5000
#指的是建立連接后從服務(wù)器讀取到可用資源所用的時(shí)間
ConnectTimeout: 5000
OpenFeign日志打印
OpenFeign提供了日志打印的功能,我們可以通過日志的打印監(jiān)控接口的調(diào)用情況,從而了解接口調(diào)用時(shí)HTTP請求的具體細(xì)節(jié),具體的使用分
第一步: 使用配置類配置日志級別
@Configuration public class FeignConfig { /** * 日志級別 * NONE:默認(rèn)的,不顯示任何日志 * BASIC:僅記錄請求方法、URL、響應(yīng)狀態(tài)碼以及執(zhí)行時(shí)間 * HEADERS:請求方法、URL、響應(yīng)狀態(tài)碼、執(zhí)行時(shí)間、請求和響應(yīng)的頭信息 * FULL:請求方法、URL、響應(yīng)狀態(tài)碼、執(zhí)行時(shí)間、請求和響應(yīng)的頭信息、正文以及元數(shù)據(jù) */ @Bean Logger.Level feignLoggerLevel() { return Logger.Level.FULL; } }
第二步: 配置文件開啟日志打印,并指定監(jiān)控的接口和級別
logging:
level:
# feign日志以 debug 級別監(jiān)控 com.atguigu.springcloud.service.PaymentFeignService 接口
com.xiaochen.springcloud.service.PaymentFeignService: debug
打印出來的日志如下:
到此這篇關(guān)于SpringCloud Ribbon與OpenFeign詳解如何實(shí)現(xiàn)服務(wù)調(diào)用的文章就介紹到這了,更多相關(guān)SpringCloud Ribbon與OpenFeign內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java實(shí)現(xiàn)word文檔轉(zhuǎn)pdf并添加水印的方法詳解
這篇文章主要介紹了java實(shí)現(xiàn)word文檔轉(zhuǎn)pdf并添加水印的方法,結(jié)合實(shí)例形式詳細(xì)分析了java word文檔轉(zhuǎn)PDF相關(guān)實(shí)現(xiàn)技巧與操作注意事項(xiàng),需要的朋友可以參考下2019-09-09單點(diǎn)登錄的概念及SpringBoot實(shí)現(xiàn)單點(diǎn)登錄的操作方法
在本文中,我們將使用Spring Boot構(gòu)建一個(gè)基本的單點(diǎn)登錄系統(tǒng),我們將介紹如何使用Spring Security和JSON Web Tokens(JWTs)來實(shí)現(xiàn)單點(diǎn)登錄功能,本文假設(shè)您已經(jīng)熟悉Spring Boot和Spring Security,感興趣的朋友一起看看吧2024-10-10ActiveMQ消息隊(duì)列技術(shù)融合Spring過程解析
這篇文章主要介紹了ActiveMQ消息隊(duì)列技術(shù)融合Spring過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-11-11JAVA實(shí)戰(zhàn)項(xiàng)目實(shí)現(xiàn)客戶選購系統(tǒng)詳細(xì)流程
讀萬卷書不如行萬里路,只學(xué)書上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實(shí)戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用Java實(shí)現(xiàn)一個(gè)簡單的客戶選購系統(tǒng),大家可以在過程中查缺補(bǔ)漏,提升水平2021-10-10如何利用Spring把元素解析成BeanDefinition對象
這篇文章主要介紹了如何利用Spring把元素解析成BeanDefinition對象,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-08-08詳解Spring中@Autowired注解是如何實(shí)現(xiàn)的
在使用java?config的過程當(dāng)中,我們不可避免地會有各種各樣的注解打交道,其中,我們使用最多的注解應(yīng)該就是@Autowired注解了,這篇文章就來和大家聊聊它到底怎么實(shí)現(xiàn)的吧2023-07-07Java8如何利用Lambda快速生成map、多層嵌套map
這篇文章主要介紹了Java8如何利用Lambda快速生成map、多層嵌套map問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-09-09