淺談Spring 5的響應式編程

来源:https://www.cnblogs.com/youruike-/archive/2020/02/04/12260132.html
-Advertisement-
Play Games

這篇使用Spring 5進行響應式編程的入門文章展示了你現在可以使用的一些新的non-blocking, asynchronous。感謝優銳課老師給予的指導! 近年來,由於響應式編程能夠以聲明性的方式(而不是強制性的)構建應用程式,從而在響應程式和彈性方面具有更強的響應能力,因此在開發人員社區和客戶 ...


這篇使用Spring 5進行響應式編程的入門文章展示了你現在可以使用的一些新的non-blocking, asynchronous。感謝優銳課老師給予的指導!

近年來,由於響應式編程能夠以聲明性的方式(而不是強制性的)構建應用程式,從而在響應程式和彈性方面具有更強的響應能力,因此在開發人員社區和客戶中日益流行。Spring 5將Reactive Systems納入其核心框架的事實表明,範式已向聲明式編程轉移。

響應式編程管理數據生產者與需要以非阻塞方式對數據做出反應的使用者之間的非同步數據流。因此,響應式編程全部與非同步和事件驅動的非阻塞應用程式有關,這些應用程式需要少量線程來擴展。

由於基於共用的可變狀態,線程和鎖擴展應用程式存在很高的複雜性,因此很難使用基於線程的框架來構建反應性應用程式。

在響應式編程上下文中,“一切都是流,並且當流中有數據時,一切都以非阻塞的方式進行。”

 

為什麼響應式編程

響應式編程的高度抽象性提高了代碼的可讀性,因此開發人員可以主要關註定義業務邏輯的事件的相互依賴性。

反應模式自然適合高度併發環境中的消息處理,這是企業常見的用例。

具有強制背壓的功能,響應式方法最適合控制生產者和消費者之間的流量,這將有助於避免記憶體不足的問題。

響應式編程可以更有效地管理高度互動和實時的應用程式或任何動作/事件可能觸發多個連接子系統的通知的情況。

 

實現響應式編程的理想用例

  • 大量交易處理服務,例如銀行業。
  • 大型線上購物應用程式(例如Amazon)的通知服務。
  • 股票交易同時變化的股票交易業務。

響應式流

“響應流”定義了一個API規範,該規範包含一組最少的介面,這些介面公開了用於定義具有非阻塞背壓的非同步數據流的操作和實體的方法。

引入反壓後,反應流允許訂戶控制發佈者的數據交換速率。

Reactive Streams API作為java.util.concurrent.Flow正式成為Java 9的一部分。

響應式流主要用作互操作性層。

 

Spring 5響應式編程產品

Spring-Web-Reactive模塊和Spring MVC都支持相同的@Controller編程,但是Spring-Web-Reactive另外在Reactive和非阻塞引擎上執行。

Spring-Web-Reactive模塊和Spring MVC共用許多常用演算法,但是Spring-Web-Reactive模塊已經重新定義了許多Spring MVC合約,例如HandlerMapping和HandlerAdapter,以使它們非同步和非阻塞並啟用 反應性HTTP請求和響應(以RouterFunction和HandlerFunction的形式)。

除了現有的RestTemplate之外,Spring 5中還引入了新的反應式WebClient。

支持響應式編程的HTTP客戶端(例如Reactor,Netty,Undertow)已經適應了一組響應式的ClientHttpRequest和ClientHttpResponse抽象,這些抽象將請求和響應主體公開為Flux <DataBuffer>,並且在讀取和寫入端具有完全的反壓支持。

Spring 5 Framework引入了Reactor作為Reactive Streams規範的實現。

Reactor是下一代Reactive庫,用於在JVM上構建非阻塞應用程式。

Reactor擴展了基本的Reactive Streams Publisher合同,並定義了Flux和Mono API類型,以分別對0..N和0..1的數據序列提供聲明性操作。

Spring Web Reactive利用Servlet 3.1提供的非阻塞I / O併在Servlet 3.1容器上運行。

Spring WebFlux提供了兩種編程模型的選擇。

  1. 帶註釋的控制器:這些與Spring MVC相同,帶有一些Spring-Web模塊提供的附加註釋。Spring MVC和WebFlux控制器都支持Reactive返回類型。此外,WebFlux還支持Reactive @RequestBody參數。
  2. 函數式編程模型:一個基於lambda的輕量級小型庫,它公開實用程式來路由和處理請求。

 

Spring Web響應式與Spring Web MVC

Spring 5彼此相鄰容納了Spring Web Reactive(在spring-web-reactive模塊下)和Spring Web MVC(在spring-webmvc模塊下)。

儘管Spring Web Reactive和Spring Web MVC模塊都共用許多演算法,但是由於Spring Web Reactive能夠在Reactive和非阻塞的Reactive Streams HTTP適配器層上運行,因此它們不共用代碼。

Spring MVC執行需要Servlet容器,而Spring Web Reactive也可以在非Servlet運行時上運行,例如Netty和Undertow。

如果絕對需要帶有Java 8 lambda或Kotlin的輕量級功能性Web框架的無阻塞Web堆棧,則應考慮從Spring MVC應用程式切換到Spring Web Reactive。

響應式編程的基本配置

這是帶有5.0.0 M5版本和WebFlux依賴項的pom.xml。

 1 <parent>
 2     <groupId>org.springframework.boot</groupId>
 3     <artifactId>spring-boot-starter-parent</artifactId>
 4     <version>2.0.0.M5</version>
 5 </parent>
 6 <dependencies>         
 7     <dependency>
 8         <groupId>org.springframework.boot</groupId>
 9         <artifactId>spring-boot-starter-webflux</artifactId>                               </dependency>
10 </dependencies>

 

傳統方法與反應方法

在傳統方法中,執行將被阻止,並將一直等到服務執行完成。在下麵的代碼中,在第一個列印語句之後,程式執行將被阻塞並等待服務執行完成。服務執行完成後,將恢復程式執行並執行第二個列印語句。

1 @GetMapping("/traditional")
2 public List < Product > getAllProducts() {
3     System.out.println("Traditional way started");
4     List < Product > products = prodService.getProducts("traditional");
5     System.out.println("Traditional way completed");
6     return products;
7 }

      

在響應式方法中,程式將繼續執行,而無需等待服務執行的完成。 在下麵的代碼中,在第一個列印語句之後,第二個列印語句將以非阻塞方式執行,而無需等待服務執行完成。將使用產品數據填充Flux流。

1 @GetMapping(value = "/reactive", .TEXT_EVENT_STREAM_VALUE)
2 public Flux < Product > getAll() {
3     System.out.println("Reactive way using Flux started");
4     Flux < Product > fluxProducts = prodService.getProductsStream("Flux");
5     System.out.println("Reactive way using Flux completed");
6     return fluxProducts;
7 }

 

響應式Web客戶端

除了現有的RestTemplate之外,Spring 5還引入了Reactive WebClient。

ClientHttpRequest和ClientHttpResponse抽象將請求和響應主體公開為Flux <DataBuffer>,在讀取和寫入側具有完全的反壓支持。

來自Spring Core的Encoder和Decoder抽象也用於客戶端,用於與類型對象之間的位元組通量序列化。

以下是一個Reactive WebClient的示例,該示例調用終結點並接收和處理Reactive Stream Flux對象。

1 @GetMapping("/accounts/{id}/alerts")
2 public Flux < Alert > getAccountAlerts(@PathVariable Long id) {
3     WebClient webClient = new WebClient(new ReactorClientHttpConnector());
4     return this.repository.getAccount(id).flatMap(account -> webClient.perform(get("/alerts/{key}", account.getKey())).extract(bodyStream(Alert.class)));
5 }

 

Spring 5的局限性

  • 對響應式應用程式進行故障排除有些困難,並且有可能在解決問題時偶然引入了阻止代碼。
  • 大多數傳統的基於Java的集成庫仍處於阻塞狀態。
  • 除少數NoSQL資料庫(例如MongoDB)外,Reactive數據存儲區提供有限的選項。
  • 仍然不支持Spring Security。

感謝閱讀!

另外近期整理了一套完整的java架構思維導圖,分享給同樣正在認真學習的每位朋友~


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本次講解函數,由於內容比較多,小編列了個大綱,主要有一下內容: 1. 函數基本語法及特性 2. 函數參數 3.局部變數 4. 返回值 5.嵌套函數 6.遞歸 7.匿名函數 8.高階函數 9.內置函數 1. 函數基本語法及特性 函數的定義:函數是指將一組語句的集合通過一個名字(函數名)封裝起來,要想執 ...
  • 我們都知道,語言嘛,當然是用來溝通交流、傳遞信息的,在人類的世界里,語言的種類非常多,像漢語、英語、日語、法語等等。 所以,我們不妨猜想一下,在電腦的世界里,會不會也是這樣的呢?答案是,的確如此。 為了使電腦能夠理解人的意圖,人類就必須將需解決的問題的思路、方法和手段通過電腦能夠理解的形式告訴... ...
  • 應用 MainModule 正確的方法是將連接組件放置在 MainModule 上, 並將數據集放在窗體或窗體所擁有的 DataModules 上。 MainModule 可以被視為會話的核心。 它是一個特殊用途 DataModule, 它是在每次創建新項目時自動創建並添加到項目中的。 MainMo ...
  • 下圖表示 uniGUI 伺服器的內部結構。 每個 uniGUI 伺服器都有一個ServerModule的副本, 每台伺服器創建一次, 同時根據用戶活動動態創建和銷毀多個Session。 uniGUI Session包含一個稱為MainModule(多個)的專用DataModule , 它為每個Ses ...
  • 在UniGUI的CHM幫助里讀到的。 一定要新建一個其他空白的工程,然後再添加LoginForm LoginForm 是另一種特殊的表單類型, 僅用於登錄目的。 此操作將創建一個與常規窗體外觀相同的空白 LoginForm: LoginForm 是一個名為 TUniLoginForm 的內置類的後代 ...
  • Dubbo是一個遠程調用的框架,對於一個服務提供者,暴露了一個介面供外部消費者調用, 那麼對於提供者自己是否可以調用這個介面,需要什麼特殊處理嗎? 這篇文章就分享下Dubbo關於本地調用的實現機制,以及如何開啟和關閉本地調用。 injvm支持本地調用 使用 Dubbo 本地調用不需做特殊配置,按正常 ...
  • Mixins Mixins(混入功能)相當於多繼承,也就是說可以繼承多個類,使用with關鍵字來實現Mixins的功能。 那麼多個類中有相同的方法時候,會被覆蓋嗎?覆蓋的先後是什麼? class A{ void a(){ print("A.a()..."); } } class B{ void a( ...
  • Dart作為一種高級語言,支持面向對象的很多特性,並且支持基於mixin的繼承方式。 基於mixin的繼承方式是指:一個類可以繼承自多個父類,相當於其他語言里的多繼承。 所有的類都有同一個基類Object,這和特性類似於Java、Objective-C 等語言,Java所有的類也都是繼承自Objec ...
一周排行
    -Advertisement-
    Play Games
  • C#TMS系統代碼-基礎頁面BaseCity學習 本人純新手,剛進公司跟領導報道,我說我是java全棧,他問我會不會C#,我說大學學過,他說這個TMS系統就給你來管了。外包已經把代碼給我了,這幾天先把增刪改查的代碼背一下,說不定後面就要趕鴨子上架了 Service頁面 //using => impo ...
  • 委托與事件 委托 委托的定義 委托是C#中的一種類型,用於存儲對方法的引用。它允許將方法作為參數傳遞給其他方法,實現回調、事件處理和動態調用等功能。通俗來講,就是委托包含方法的記憶體地址,方法匹配與委托相同的簽名,因此通過使用正確的參數類型來調用方法。 委托的特性 引用方法:委托允許存儲對方法的引用, ...
  • 前言 這幾天閑來沒事看看ABP vNext的文檔和源碼,關於關於依賴註入(屬性註入)這塊兒產生了興趣。 我們都知道。Volo.ABP 依賴註入容器使用了第三方組件Autofac實現的。有三種註入方式,構造函數註入和方法註入和屬性註入。 ABP的屬性註入原則參考如下: 這時候我就開始疑惑了,因為我知道 ...
  • C#TMS系統代碼-業務頁面ShippingNotice學習 學一個業務頁面,ok,領導開完會就被裁掉了,很突然啊,他收拾東西的時候我還以為他要旅游提前請假了,還在尋思為什麼回家連自己買的幾箱飲料都要叫跑腿帶走,怕被偷嗎?還好我在他開會之前拿了兩瓶芬達 感覺感覺前面的BaseCity差不太多,這邊的 ...
  • 概述:在C#中,通過`Expression`類、`AndAlso`和`OrElse`方法可組合兩個`Expression<Func<T, bool>>`,實現多條件動態查詢。通過創建表達式樹,可輕鬆構建複雜的查詢條件。 在C#中,可以使用AndAlso和OrElse方法組合兩個Expression< ...
  • 閑來無聊在我的Biwen.QuickApi中實現一下極簡的事件匯流排,其實代碼還是蠻簡單的,對於初學者可能有些幫助 就貼出來,有什麼不足的地方也歡迎板磚交流~ 首先定義一個事件約定的空介面 public interface IEvent{} 然後定義事件訂閱者介面 public interface I ...
  • 1. 案例 成某三甲醫預約系統, 該項目在2024年初進行上線測試,在正常運行了兩天後,業務系統報錯:The connection pool has been exhausted, either raise MaxPoolSize (currently 800) or Timeout (curren ...
  • 背景 我們有些工具在 Web 版中已經有了很好的實踐,而在 WPF 中重新開發也是一種費時費力的操作,那麼直接集成則是最省事省力的方法了。 思路解釋 為什麼要使用 WPF?莫問為什麼,老 C# 開發的堅持,另外因為 Windows 上已經裝了 Webview2/edge 整體打包比 electron ...
  • EDP是一套集組織架構,許可權框架【功能許可權,操作許可權,數據訪問許可權,WebApi許可權】,自動化日誌,動態Interface,WebApi管理等基礎功能於一體的,基於.net的企業應用開發框架。通過友好的編碼方式實現數據行、列許可權的管控。 ...
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...