Spring Cloud構建微服務架構:消息驅動的微服務(入門)【Dalston版】

2019獨角獸企業重金招聘Python工程師標準>>> hot3.png

之前在寫Spring Boot基礎教程的時候寫過一篇《Spring Boot中使用RabbitMQ》。在該文中,我們通過簡單的配置和注解就能實現向RabbitMQ中生產和消費消息。實際上我們使用的對RabbitMQ的starter就是通過Spring Cloud Stream中對RabbitMQ的支持來實現的。下面我們就通過本文來了解一下Spring Cloud Stream。

Spring Cloud Stream是一個用來為微服務應用構建消息驅動能力的框架。它可以基于Spring Boot來創建獨立的、可用于生產的Spring應用程序。它通過使用Spring Integration來連接消息代理中間件以實現消息事件驅動的微服務應用。Spring Cloud Stream為一些供應商的消息中間件產品提供了個性化的自動化配置實現,并且引入了發布-訂閱、消費組以及消息分區這三個核心概念。簡單的說,Spring Cloud Stream本質上就是整合了Spring Boot和Spring Integration,實現了一套輕量級的消息驅動的微服務框架。通過使用Spring Cloud Stream,可以有效地簡化開發人員對消息中間件的使用復雜度,讓系統開發人員可以有更多的精力關注于核心業務邏輯的處理。由于Spring Cloud Stream基于Spring Boot實現,所以它秉承了Spring Boot的優點,實現了自動化配置的功能幫忙我們可以快速的上手使用,但是目前為止Spring Cloud Stream只支持下面兩個著名的消息中間件的自動化配置:

  • RabbitMQ
  • Kafka

快速入門

下面我們通過構建一個簡單的示例來對Spring Cloud Stream有一個初步認識。該示例主要目標是構建一個基于Spring Boot的微服務應用,這個微服務應用將通過使用消息中間件RabbitMQ來接收消息并將消息打印到日志中。所以,在進行下面步驟之前請先確認已經在本地安裝了RabbitMQ,具體安裝步驟請參考此文。

構建一個Spring Cloud Stream消費者

  • 創建一個基礎的Spring Boot工程,命名為:stream-hello

  • 編輯pom.xml中的依賴關系,引入Spring Cloud Stream對RabbitMQ的支持,具體如下:

<parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>1.5.9.RELEASE</version><relativePath/> <!-- lookup parent from repository -->
</parent><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-stream-rabbit</artifactId>     </dependency>
</dependencies><dependencyManagement><dependencies><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-dependencies</artifactId><version>Dalston.SR4</version><type>pom</type><scope>import</scope></dependency></dependencies>
</dependencyManagement>
  • 創建用于接收來自RabbitMQ消息的消費者SinkReceiver,具體如下:
@EnableBinding(Sink.class)
public class SinkReceiver {private static Logger logger = LoggerFactory.getLogger(SinkReceiver.class);@StreamListener(Sink.INPUT)public void receive(Object payload) {logger.info("Received: " + payload);}}
  • 創建應用主類,這里同其他Spring Boot一樣,沒有什么特別之處,具體如下:
@SpringBootApplication
public class SinkApplication {public static void main(String[] args) {SpringApplication.run(SinkApplication.class, args);}}

到這里,我們快速入門示例的編碼任務就已經完成了。下面我們分別啟動RabbitMQ以及該Spring Boot應用,然后做下面的試驗,看看它們是如何運作的。

手工測試驗證

  • 我們先來看一下Spring Boot應用的啟動日志。
...
INFO 16272 --- [main] o.s.c.s.b.r.RabbitMessageChannelBinder   : declaring queue for inbound: input.anonymous.Y8VsFILmSC27eS5StsXp6A, bound to: input
INFO 16272 --- [main] o.s.a.r.c.CachingConnectionFactory       : Created new connection: SimpleConnection@3c78e551 [delegate=amqp://guest@127.0.0.1:5672/]
INFO 16272 --- [main] o.s.integration.channel.DirectChannel    : Channel 'input.anonymous.Y8VsFILmSC27eS5StsXp6A.bridge' has 1 subscriber(s).
INFO 16272 --- [main] o.s.i.a.i.AmqpInboundChannelAdapter      : started inbound.input.anonymous.Y8VsFILmSC27eS5StsXp6A
...

從上面的日志內容中,我們可以獲得以下信息:

  • 使用guest用戶創建了一個指向127.0.0.1:5672位置的RabbitMQ連接,在RabbitMQ的控制臺中我們也可以發現它。

  • 聲明了一個名為input.anonymous.Y8VsFILmSC27eS5StsXp6A的隊列,并通過RabbitMessageChannelBinder將自己綁定為它的消費者。這些信息我們也能在RabbitMQ的控制臺中發現它們。

下面我們可以在RabbitMQ的控制臺中進入input.anonymous.Y8VsFILmSC27eS5StsXp6A隊列的管理頁面,通過Publish Message功能來發送一條消息到該隊列中。

此時,我們可以在當前啟動的Spring Boot應用程序的控制臺中看到下面的內容:

INFO 16272 --- [C27eS5StsXp6A-1] com.didispace.HelloApplication           : Received: [B@7cba610e

我們可以發現在應用控制臺中輸出的內容就是SinkReceiverreceive方法定義的,而輸出的具體內容則是來自消息隊列中獲取的對象。這里由于我們沒有對消息進行序列化,所以輸出的只是該對象的引用,在后面的小節中我們會詳細介紹接收消息后的處理。

在順利完成上面快速入門的示例后,我們簡單解釋一下上面的步驟是如何將我們的Spring Boot應用連接上RabbitMQ來消費消息以實現消息驅動業務邏輯的。

首先,我們對Spring Boot應用做的就是引入spring-cloud-starter-stream-rabbit依賴,該依賴包是Spring Cloud Stream對RabbitMQ支持的封裝,其中包含了對RabbitMQ的自動化配置等內容。從下面它定義的依賴關系中,我們還可以知道它等價于spring-cloud-stream-binder-rabbit依賴。

<dependencies><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-stream-binder-rabbit</artifactId></dependency>
</dependencies>

接著,我們再來看看這里用到的幾個Spring Cloud Stream的核心注解,它們都被定義在SinkReceiver中:

  • @EnableBinding,該注解用來指定一個或多個定義了@Input@Output注解的接口,以此實現對消息通道(Channel)的綁定。在上面的例子中,我們通過@EnableBinding(Sink.class)綁定了Sink接口,該接口是Spring Cloud Stream中默認實現的對輸入消息通道綁定的定義,它的源碼如下:
public interface Sink {String INPUT = "input";@Input(Sink.INPUT)SubscribableChannel input();}

它通過@Input注解綁定了一個名為input的通道。除了Sink之外,Spring Cloud Stream還默認實現了綁定output通道的Source接口,還有結合了SinkSourceProcessor接口,實際使用時我們也可以自己通過@Input@Output注解來定義綁定消息通道的接口。當我們需要為@EnableBinding指定多個接口來綁定消息通道的時候,可以這樣定義:@EnableBinding(value = {Sink.class, Source.class})

  • @StreamListener:該注解主要定義在方法上,作用是將被修飾的方法注冊為消息中間件上數據流的事件監聽器,注解中的屬性值對應了監聽的消息通道名。在上面的例子中,我們通過@StreamListener(Sink.INPUT)注解將receive方法注冊為對input消息通道的監聽處理器,所以當我們在RabbitMQ的控制頁面中發布消息的時候,receive方法會做出對應的響應動作。

編寫消費消息的單元測試用例

上面我們通過RabbitMQ的控制臺完成了發送消息來驗證了消息消費程序的功能,雖然這種方法比較low,但是通過上面的步驟,相信大家對RabbitMQ和Spring Cloud Stream的消息消費已經有了一些基礎的認識。下面我們通過編寫生產消息的單元測試用例來完善我們的入門內容。

  • 在上面創建的工程中創建單元測試類:
@RunWith(SpringRunner.class)
@EnableBinding(value = {SinkApplicationTests.SinkSender.class})
public class SinkApplicationTests {@Autowiredprivate SinkSender sinkSender;@Testpublic void sinkSenderTester() {sinkSender.output().send(MessageBuilder.withPayload("produce a message :http://blog.didispace.com").build());}public interface SinkSender {String OUTPUT = "input";@Output(SinkSender.OUTPUT)MessageChannel output();}}
  • 在應用了上面的消息消費者程序之后,運行這里定義的單元測試程序,我們馬上就能在消息消費者的控制臺中收到下面的內容:
INFO 50947 --- [L2W-c2AcChb2Q-1] com.didispace.stream.SinkReceiver        : Received: produce a message :http://blog.didispace.com

在上面的單元測試中,我們通過@Output(SinkSender.OUTPUT)定義了一個輸出通過,而該輸出通道的名稱為input,與前文中的Sink中定義的消費通道同名,所以這里的單元測試與前文的消費者程序組成了一對生產者與消費者。到這里,本文的內容就次結束,如果您能夠獨立的完成上面的例子,那么對于Spring Cloud Stream的基礎使用算是入門了。但是,Spring Cloud Stream的使用遠不止于此,在近期的博文中,我講繼續更新這部分內容,幫助他們來理解和用好Spring Cloud Stream來構建消息驅動的微服務!

本文完整實例:

  • Github
  • Gitee

如果您對這些感興趣,歡迎star、follow、收藏、轉發給予支持!

本文內容部分節選自我的《Spring Cloud微服務實戰》,但對依賴的Spring Boot和Spring Cloud版本做了升級。

本文首發于我的博客:http://blog.didispace.com

系列教程推薦

  • Spring Boot基礎教程
  • Spring Cloud基礎教程

轉載于:https://my.oschina.net/didispace/blog/1592356

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/255936.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/255936.shtml
英文地址,請注明出處:http://en.pswp.cn/news/255936.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

CXF 客服端調用報錯

服務端已經發布了WSDL&#xff0c;現在在客服端生成web service客服端代碼&#xff0c;在eclipse中新建一個project&#xff0c;然后new->web services->web service client生產客戶端代碼 在調用的時候報如下錯誤 解決&#xff1a;缺少axis相應的jar包&#xff0c;加入包…

20145225 《信息安全系統設計基礎》第10周學習總結

cp1.c 進行復制文件的操作&#xff0c;需要有源文件和目的文件&#xff0c;第一次命令沒有加入所以沒有正常完成復制文件的操作fileinfo.c 用來實現顯示文件信息。先判斷命令是否有操作數&#xff0c;有的話才能繼續進行下去&#xff0c;如果沒有報錯就打印出來相關文件信息&am…

做演員是圓夢 做生意學會面對現實

田樸珺是一位擁有多重身份的女性。她是一名演員&#xff0c;也是一位商人&#xff0c;還擔任過電影《中國合伙人》的制片人。 作為演員&#xff0c;田樸珺的作品并不是很多&#xff0c;也一直不溫不 火。但這并不代表她將放棄演藝生涯。她表示&#xff0c;如果機會合適&…

【深度學習】——模型評估指標MAP計算實例計算

目錄 一、知識儲備 1、IOU——交集面積與并集面積之比 2、混淆矩陣&#xff08;TP、FP、FN、TN&#xff09; 問題1&#xff1a;上面的TP等具體是如何計算得到的&#xff1f; 3、精度precision&召回率recall 二、ap計算實戰 1、計算流程 1&#xff09;準備數據&#xf…

第 52 章 Web Server Optimization

系統配置 Intel(R) Xeon(TM) CPU 3.00GHzMemory 4GEthernet adapter 1000M52.1. ulimit 查看 ulimit ulimit -a core file size (blocks, -c) 0 data seg size (kbytes, -d) unlimited file size (blocks, -f) unlimited pending signals …

hdu5489 Removed Interval dp+線段樹優化

現在看這題居然直接秒了。。。去年看的時候還以為神題。。 設以第i項為結尾的lis前綴為f[i]&#xff0c;以第j項為結尾的lis后綴為g[i]&#xff0c;如果求出f[i]和g[j]&#xff0c;然后枚舉i&#xff0c;快速找到最大的滿足a[j]>a[i]的g[j]就可以了。注意到如果將f[i]從后往…

JS原型鏈理解

1. 每個對象都有原型屬性(__proto__)2. 對象的原型(__proto__)指向其構造函數(Constructor)的prototype屬性3. 構造函數(Constructor)的prototype屬性本身也是一個對象&#xff0c;其原型(__proto__)亦指向其構造函數的prototype4. 如此形成一個鏈式結構&#xff0c;而Construc…

【深度學習】——2021年FPN特征金字塔

#!/usr/bin/env python # -*- coding: utf-8 -*- # Time : 2021/4/22 17:06 # Author : linlianqin # Site : # File : fpn.py # Software: PyCharm # description:其搭建的基本流程和resnet是一致的&#xff0c;只是將每一層的卷積結果保存了起來import torch impo…

NoSQL分類及ehcache memcache redis 三大緩存的對比

NoSQL分類 由于NoSQL中沒有像傳統數據庫那樣定義數據的組織方式為關系型的&#xff0c;所以只要內部的數據組織采用了非關系型的方式&#xff0c;就可以稱之為NoSQL數據庫。目前&#xff0c;可以將眾多的NoSQL數據庫按照內部的數據組織形式進行如下分類&#xff1a; Key/Value的…

52.4. APC Cache (php-apc - APC (Alternative PHP Cache) module for PHP 5)

$ apt-cache search php-apc php-apc - APC (Alternative PHP Cache) module for PHP 5$ sudo apt-get install php-apcapc cache 狀態監控 http://pecl.php.net/package/APC 下載解包找到apc.php,放到web服務器上 原文出處&#xff1a;Netkiller 系列 手札 本文作者&#xff1…

樂視云計算基于OpenStack的IaaS實踐

本文作者岳龍廣&#xff0c;現在就職于樂視云計算有限公司&#xff0c;負責IaaS部門的工作。 從開始工作就混在開源世界里&#xff0c;在虛擬化方面做過CloudStack/Ovirt開發&#xff0c;現在是做以OpenStack為基礎的樂視云平臺。所以對虛擬化情有獨鐘&#xff0c;也對虛擬化/云…

【深度學習】——如何提高map值

目錄 代碼獲取 map原理 map提高技巧 技巧總結&#xff1a; 實戰&#xff1a; 1、效果不佳map55.55% 1&#xff09;單獨調整get_dr_txt.py中的self.iou 0.3 2&#xff09;單獨調整get_map,py中的minoverlap: 3)同時調整minoverlap和self.iou 本文是在faster_rcnn模型的…

每日站立會議個人博客(沖刺周)-Wednesday

時間未完成不知道如何獲取具體標簽里的內容正在做爬蟲技術之獲取標簽里的內容將要做對運用爬蟲技術獲取的數據進行處理轉載于:https://www.cnblogs.com/andibier/p/8075098.html

數據庫水平切分的實現原理解析——分庫,分表,主從,集群,負載均衡器(轉)...

第1章 引言 隨著互聯網應用的廣泛普及&#xff0c;海量數據的存儲和訪問成為了系統設計的瓶頸問題。對于一個大型的互聯網應用&#xff0c;每天幾十億的PV無疑對數據庫造成了相當高的負載。對于系統的穩定性和擴展性造成了極大的問題。通過數據切分來提高網站性能&#xff0c;橫…

【深度學習】——糾錯error: Unable to find vcvarsall.bat:關于安裝pycocotools

1、安裝包下載 大佬改寫支持 Windows 的 COCO 地址&#xff1a;https://github.com/philferriere/cocoapi 下載后如下&#xff1a; 進入pythonAPI 先后運行&#xff1a; python setup.py build_ext --inplacepython setup.py build_ext install 出現以下標志時&#xff0c…

【小貼士】虛擬鍵盤與fixed帶給移動端的痛!

前言今天來公司的主要目的就是研究虛擬鍵盤與fixed的問題&#xff0c;期間因為同事問起閉包與事件委托&#xff08;阻止冒泡&#xff09;相關問題&#xff0c;便穿插了一篇別的&#xff1a;【小貼士】工作中的”閉包“與事件委托的”阻止冒泡“&#xff0c;有興趣的朋友可以去看…

[OJ] Wildcard Matching (Hard)

LintCode 192. Wildcard Matching (Hard)LeetCode 44. Wildcard Matching (Hard) 第二次刷還是被這題虐. 其實就是跪在一個地方, 就是關于mustFail的地方. 當*p && !*s的時候, 說明s已經被用完了, p還沒有被窮盡, 這種情況下要直接退出所有的遞歸返回false, 因為s都匹配…

CSS3 -webkit-transition(屬性漸變)

-webkit-transition&#xff1a;CSS屬性(none|all|屬性) 持續時間 時間函數 延遲時間 CSS屬性(transition-property)&#xff1a;要變化的屬性&#xff0c;比如元素變寬則是width&#xff0c;文字顏色要變色這是color&#xff1b;W3C給出了一個可變換屬性的列表&#xff1a;…

vxworks的default boot line說明

boot程序的主要功能是引導vxworks 內核,所以boot程序需要知道vxworks的內核存放在何處&#xff0c;通過什么手段去獲取。在vxworks缺省的boot程序里有一條內建的default boot line,它指明了獲得vxworks內核的途徑&#xff0c;在boot程序啟動時&#xff0c;它先尋找NVRAM里面有無…

【機器視覺】——相機和鏡頭的選擇

目錄 1、相機選擇 2、鏡頭選擇 3、其他計算公式 1)芯片尺寸計算: