欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring Cloud Stream微服務(wù)消息框架原理及實(shí)例解析

 更新時間:2020年06月02日 09:54:17   作者:Freshchen''s Blog  
這篇文章主要介紹了Spring Cloud Stream微服務(wù)消息框架原理及實(shí)例解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下

隨著近些年微服務(wù)在國內(nèi)的盛行,消息驅(qū)動被提到的越來越多。主要原因是系統(tǒng)被拆分成多個模塊后,一個業(yè)務(wù)往往需要在多個服務(wù)間相互調(diào)用,不管是采用HTTP還是RPC都是同步的,不可避免快等慢的情況發(fā)生,系統(tǒng)性能上很容易遇到瓶頸。在這樣的背景下,將業(yè)務(wù)中實(shí)時性要求不是特別高且非主干的部分放到消息隊(duì)列中是很好的選擇,達(dá)到了異步解耦的效果。

目前消息隊(duì)列有很多優(yōu)秀的中間件,目前使用較多的主要有 RabbitMQ,Kafka,RocketMQ 等,這些中間件各有優(yōu)勢,有的對 AMQP(應(yīng)用層標(biāo)準(zhǔn)高級消息隊(duì)列協(xié)議)支持完善,有的提供了更高的可靠性,有的對大數(shù)據(jù)支持良好,同時各種消息中間件概念不統(tǒng)一,使得選擇和使用一款合適的消息中間件成為難題。Spring跳出來給出了解決方案:Spring Cloud Stream,使用它可以很方便高效的操作消息中間件,程序員只要關(guān)心業(yè)務(wù)代碼即可,目前官方支持 RabbitMQ,Kafka兩大主流MQ,RocketMQ 則自己提供了相應(yīng)支持。

首先看一下Spring Cloud Stream做了什么,如下圖所示,框架目前官方把消息中間件抽象成了 Binder,業(yè)務(wù)代碼通過進(jìn)出管道連接 Binder,各消息中間件的差異性統(tǒng)一交給了框架處理,程序員只需要了解框架的抽象出來的一些統(tǒng)一概念即可

  • Binder(綁定器):RabbitMQ,Kafka等中間件服務(wù)的封裝
  • Channel(管道):也就是圖中的 inputs 和 outputs 所指區(qū)域,是應(yīng)用程序和 Binder 的橋梁
  • Gourp(消費(fèi)組):由于微服務(wù)會部署多實(shí)例,為了保證只被服務(wù)的一個實(shí)例消費(fèi),可以通過配置,把實(shí)例都綁到同一個消費(fèi)組
  • Partitioning (消息分區(qū)):如果某一類消息只想指定給服務(wù)的固定實(shí)例消費(fèi),可以使用分區(qū)實(shí)現(xiàn)

Spring Cloud Stream將業(yè)務(wù)代碼和消息中間件解耦,帶來的好處可以從下圖很直觀的感受到,很簡潔的代碼,我們便能從RabbitMQ中接受消息然后經(jīng)過業(yè)務(wù)處理再向Kafka發(fā)送一條消息,只需要更改相關(guān)配置就能快速改變系統(tǒng)行為。

細(xì)心的讀者可能會好奇,上圖的代碼只是注入了一個簡單的 Function 而已,實(shí)際上,Spring Cloud Stream3.0后集成了Spring Cloud Function框架 ,提倡函數(shù)式的風(fēng)格,棄用先前版本基于注解的開發(fā)方式。Spring Cloud Function是 Serverless 和 Faas 的產(chǎn)物,強(qiáng)調(diào)面向函數(shù)編程,一份代碼各云平臺運(yùn)行,和Spring Cloud Stream一樣也是解決了基礎(chǔ)設(shè)施的差異性問題,通過強(qiáng)大的自動裝配機(jī)制,可以根據(jù)配置自動暴露 HTTP 服務(wù)或者消息服務(wù),并且同時支持命令式和響應(yīng)式編程模式,可以說是很強(qiáng)大了。下面通過一個簡單的例子來理解下上圖的代碼和框架的使用把。

簡單案例

模擬一個簡單的下單,收到訂單之后處理完,返回成功,然后發(fā)送消息給庫存模塊,庫存模塊再發(fā)送消息給報表模塊

項(xiàng)目地址

springcloud-stream

項(xiàng)目結(jié)構(gòu)

項(xiàng)目依賴

<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
  <groupId>org.springframework.cloud</groupId>
  <artifactId>spring-cloud-starter-stream-rabbit</artifactId>
</dependency>

表單

@Data
public class OrderForm {
  private String productName;
}

消息管道注冊

@Configuration
@Slf4j
public class MessageQueueConfig {

  @Bean
  public Function<OrderForm, OrderForm> inventory() {
    return orderForm -> {
      log.info("Inventory Received Message: " + orderForm);
      return orderForm;
    };
  }

  @Bean
  public Consumer<OrderForm> report() {
    return orderForm -> {
      log.info("Report Received Message: " + orderForm);
    };
  }
}

Controller

@Slf4j
@RestController
public class OrderController {

  @Autowired
  private BeanFactoryChannelResolver resolver;

  @PostMapping("order")
  public String order(@RequestBody OrderForm orderForm) {
    log.info("Received Request " + orderForm);
    resolver.resolveDestination("inventory-in-0").send(new GenericMessage<>(orderForm));
    return "success";
  }
}

配置

框架會按照中間件默認(rèn)端口去連接,這里自定義了一個名為myLocalRabbit的類型是RabbitMQ的Binder配置,bindings下面 inventory-in-0 是通道名,接受inventory主題(對應(yīng)RabbitMQ的ExChange)的消息,然后處理完通過 inventory-out-0 通道發(fā)送消息到 report 主題, report-in-0通道負(fù)責(zé)接受report主題的消息。

注:通道名=注冊的 function 方法名 + in或者out + 參數(shù)位置(詳見注釋)

spring:
 cloud:
  stream:
#   配置消息中間件信息
   binders:
    myLocalRabbit:
     type: rabbit
     environment:
      spring:
       rabbitmq:
        host: localhost
        port: 31003
        username: guest
        password: guest
        virtual-host: /
#   重點(diǎn),如何綁定通道,這里有個約定,開頭是函數(shù)名,in表示消費(fèi)消息,out表示生產(chǎn)消息,最后的數(shù)字是函數(shù)接受的參數(shù)的位置,destination后面為訂閱的主題
#   比如Function<Tuple2<Flux<String>, Flux<Integer>>, Flux<String>> gather()
#   gather函數(shù)接受的第一個String參數(shù)對應(yīng) gather-in-0,第二個Integer參數(shù)對應(yīng) gather-in-1,輸出對應(yīng) gather-out-0
   bindings:
    inventory-in-0:
     destination: inventory
    inventory-out-0:
     destination: report
    report-in-0:
     destination: report
#   注冊聲明的三個函數(shù)
   function:
    definition: inventory;report

測試

POST http://localhost:8080/order
Content-Type: application/json

{
 "productName": "999"
}

結(jié)果

POST http://localhost:8080/order

HTTP/1.1 200 
Content-Type: text/plain;charset=UTF-8
Content-Length: 7
Date: Sat, 30 May 2020 15:27:56 GMT
Keep-Alive: timeout=60
Connection: keep-alive

success

Response code: 200; Time: 56ms; Content length: 7 bytes

后臺日志

可以看到消息成功發(fā)送到了庫存和報表服務(wù)

2020-05-30 23:27:56.956 INFO 8760 --- [nio-8080-exec-1] c.e.springcloudstream.OrderController  : Received Request OrderForm(productName=999)
2020-05-30 23:27:56.956 INFO 8760 --- [nio-8080-exec-1] o.s.i.h.s.MessagingMethodInvokerHelper  : Overriding default instance of MessageHandlerMethodFactory with provided one.
2020-05-30 23:27:56.957 INFO 8760 --- [nio-8080-exec-1] c.e.s.MessageQueueConfig         : Inventory Received Message: OrderForm(productName=999)
2020-05-30 23:27:56.958 INFO 8760 --- [nio-8080-exec-1] o.s.a.r.c.CachingConnectionFactory    : Attempting to connect to: [localhost:31003]
2020-05-30 23:27:56.964 INFO 8760 --- [nio-8080-exec-1] o.s.a.r.c.CachingConnectionFactory    : Created new connection: rabbitConnectionFactory.publisher#6131841e:0/SimpleConnection@192fe472 [delegate=amqp://guest@127.0.0.1:31003/, localPort= 2672]
2020-05-30 23:27:56.965 INFO 8760 --- [nio-8080-exec-1] o.s.amqp.rabbit.core.RabbitAdmin     : Auto-declaring a non-durable, auto-delete, or exclusive Queue (inventory.anonymous.wtaFwHlNRkql5IUh2JCNAA) durable:false, auto-delete:true, exclusive:true. It will be redeclared if the broker stops and is restarted while the connection factory is alive, but all messages will be lost.
2020-05-30 23:27:56.965 INFO 8760 --- [nio-8080-exec-1] o.s.amqp.rabbit.core.RabbitAdmin     : Auto-declaring a non-durable, auto-delete, or exclusive Queue (report.anonymous.SJgpJKiJQf2tudszgf623w) durable:false, auto-delete:true, exclusive:true. It will be redeclared if the broker stops and is restarted while the connection factory is alive, but all messages will be lost.
2020-05-30 23:27:56.979 INFO 8760 --- [f2tudszgf623w-1] o.s.i.h.s.MessagingMethodInvokerHelper  : Overriding default instance of MessageHandlerMethodFactory with provided one.
2020-05-30 23:27:56.980 INFO 8760 --- [f2tudszgf623w-1] c.e.s.MessageQueueConfig         : Report Received Message: OrderForm(productName=999)

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Java中超詳細(xì)this與super的概念和用法

    Java中超詳細(xì)this與super的概念和用法

    這篇文章主要介紹了Java中超詳細(xì)this與super的概念和用法,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-02-02
  • WebFlux 服務(wù)編排使用優(yōu)勢詳解

    WebFlux 服務(wù)編排使用優(yōu)勢詳解

    這篇文章主要為大家介紹了WebFlux 服務(wù)編排使用優(yōu)勢示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-05-05
  • 從HelloWorld和文檔注釋開始入門Java編程

    從HelloWorld和文檔注釋開始入門Java編程

    這篇文章主要介紹了從HelloWorld和文檔注釋開始入門Java編程,涉及到Javadoc工具的使用,需要的朋友可以參考下
    2015-10-10
  • java判斷字符串是正整數(shù)的實(shí)例

    java判斷字符串是正整數(shù)的實(shí)例

    今天小編就為大家分享一篇java判斷字符串是正整數(shù)的實(shí)例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-07-07
  • java面試常見問題---ConcurrentHashMap

    java面試常見問題---ConcurrentHashMap

    ConcurrentHashMap是由Segment數(shù)組結(jié)構(gòu)和HashEntry數(shù)組結(jié)構(gòu)組成。Segment的結(jié)構(gòu)和HashMap類似,是一種數(shù)組和鏈表結(jié)構(gòu),今天給大家普及java面試常見問題---ConcurrentHashMap知識,一起看看吧
    2021-06-06
  • 什么是Spring Boot

    什么是Spring Boot

    Spring是一個非常受歡迎的Java框架,它用于構(gòu)建web和企業(yè)應(yīng)用。本文介紹將各種Spring的配置方式,幫助您了解配置Spring應(yīng)用的復(fù)雜性
    2017-08-08
  • spring boot加載資源路徑配置和classpath問題解決

    spring boot加載資源路徑配置和classpath問題解決

    這篇文章主要介紹了spring boot加載資源路徑配置和classpath問題解決,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-03-03
  • 老生常談Java網(wǎng)絡(luò)編程TCP通信(必看篇)

    老生常談Java網(wǎng)絡(luò)編程TCP通信(必看篇)

    下面小編就為大家?guī)硪黄仙U凧ava網(wǎng)絡(luò)編程TCP通信(必看篇)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-05-05
  • 詳解Java 集合系列(三)—— LinkedList

    詳解Java 集合系列(三)—— LinkedList

    這篇文章主要介紹了Java 集合系列(三)—— LinkedList,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • Spring依賴注入的三種方式小結(jié)

    Spring依賴注入的三種方式小結(jié)

    本篇文章主要介紹了Spring依賴注入的三種方式小結(jié),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-08-08

最新評論