主頁 > 後端開發 > 再見了SpringMVC,這個框架有點厲害,甚至干掉了Servlet!

再見了SpringMVC,這個框架有點厲害,甚至干掉了Servlet!

2020-11-13 14:59:27 後端開發

前言

對 Java 開發者來說, Spring 發布了 5.0 正式版后的一大特色,就是 Reactive Web 方案 Web Flux,這是用來替代 Spring Web MVC 的嗎?或者,只是終于可以不再基于 Servlet 容器了?

基于 Servlet 容器的 Web MVC

身為 Java 開發者,對于 Spring 框架并不陌生,它起源于 2002 年、Rod Johnson 著作《Expert One-on-One J2EE Design and Development》中的 Interface 21 框架,到了 2004 年,推出 Spring 1.0,從 XML 到 3.0 之后,支持 JavaConfig 設定;進一步,在 2014 年時,除了 Spring 4.0 之外,首次發表了Spring Boot,最大的亮點是采用自動組態,令基于 Spring 的快速開發成為可能,

對 Web 開發者來說,Spring 中的 Web MVC 框架,也一直隨著 Spring 而成長,然而由于基于 Servlet 容器,早期被批評不易測驗(例如:控制器中包含了 Servlet API),

不過,從實操 Controller 介面搭配 XML 設定,到后來的標注搭配 JavaConfig,Web MVC 使用越來越便利,如果愿意,也可采用漸進的方式,將基于 Servlet API 的 Web 應用程式,逐步重構為幾乎沒有 Servlet API 的存在,在程式代碼層面達到屏蔽 Servlet API 的效果,

由于不少 Java 開發者的 Web 開發經驗,都是從 Servlet 容器中累積起來的,在這個時候,Web MVC 框架基于 Servlet API,就會是一項優點,因為,雖然運用 Web MVC 撰寫程式時,可做到不直接面對 Servlet API,然而,也意味著更強烈地受到 Spring 的約束,有時則是無法在設定或 API 中找到對應方案,有時也因為心智模型還是掛在 Servlet 容器,經驗上難以脫離,在搞不出 HttpSession、ServletContext 對應功能時,直接從 HttpSession、ServletContext 下手,畢竟也是個方法,

撰寫程式時,就算沒用到 Servlet API,Web MVC 基于 Servlet 容器仍是事實,因為,底層還是得借助 Servlet 容器的功能,例如 Spring Security,本質上還是基于 Servlet 容器的 Filter 方案,

然而在今日,Servlet 被許多開發者視為陳舊、過時技術的象征,或許是因為這樣,在 Java EE 8 宣布推出的這段期間,當在某些場合談及 Servlet 4.0 之時,總會聽到有人提出“Web Flux 可以脫離 Servlet 了”之類的建議,

實作 Reactive Streams 的 Reactor

Web Flux 不依賴 Servlet 容器是事實,然而,在談及 Web Flux 之前,我們必須先知道 Reactor 專案,它是由 Pivotal 公司,也就是目前 Spring 的擁有者推出,實作了 Reactive Streams 規范,用來支持 Reactive Programming 的實作品,

既然是實作了 Reactive Streams 規范,開發者必然會想到的是 RxJava/RxJava 2,或者是 Java 9 的 Flow API,這也意謂著,在能使用 Web Flux 之前,開發者必須對于 Reactive Programming 典范,有所認識,

開發者這時有疑問了,Spring 為何不直接基于 RxJava 2,而是打造專屬的 Reactive Streams 專案呢?

就技術而言,Reactor 是在 Java 8 的基礎上開發,并全面擁抱 Java 8 之后的新 API,像是 Lambda 相關介面、新日期與時間 API 等,這意謂著,專案如果還是基于 Java 7 或更早版本,就無法使用 Reactor,

在 API 層面,RxJava 2 有著因為歷史發展脈絡的原因,不得不保留一些令人容易困惑或混淆的型態或操作,而 Reactor 在這方面,都有著明確的對應 API 來取代,然而,卻也提供與 RxJava 2(甚至是 Flow API)間的轉換,

另一方面,Reactor 較直覺易用,例如最常介紹的 Mono 與 Flux,實作了 Reactive Streams 的 Publisher界面,并簡化了資訊發布,讓開發者在許多場合,不用處理 Subscriber 和 Subscription 的細節(當然,這些在 Reactor 也予以實作),而在 Spring Web Flux 中,Mono 與 Flux 也是主要的操作物件,想知道如何使用Mono與Flux,可以參考〈使用 Reactor 進行反應式編程〉

又一個 Web 框架?

到了 Spring 5,在 Reactor 的基礎上,新增了 Web Flux 作為 Reactive Web 的方案,我們在許多介紹檔案的簡單示例,例如〈使用 Spring 5 的 WebFlux 開發反應式 Web 應用〉,就看到當中使用了 Flux、Mono 來示范,而且,程式的代碼看起來就像是 Spring MVC,

這是因為 Web Flux 提供了基于 Java 注解的方式,有許多 Web MVC 中使用的標注,也拿來用在 Web Flux 之中,讓熟悉 Web MVC 的開發者也容易理解與上手 Web Flux,然而,這不過就是新的 Web 框架嗎?

實際上,當然不是如此,Web Flux 并不依賴 Web MVC,而且它是基于 Reactor,本質屬于非同步、非阻斷、Reactive Programming 的心智模型,也因此,如果打算將 Web Flux 運行在 Servlet 容器之上,必須是支持 Servlet 3.1 以上,因為才有非阻斷輸入輸出的支持,雖然 Web Flux 的 API 在某些地方,確實提供了阻斷的選項,若單純只是試著將基于 Web MVC 的應用程式,改寫為套用 Web Flux,并不會有任何益處,反而會窮于應付如何在 Web Flux 實作對應的方案,

例如,此時,Spring Security 顯然就不能用了,畢竟是 Spring 基于 Servlet 的安全方案,開發者必須想辦法套用 Spring Security Reactive;而且,在儲存方案上,也不是直接采用 Spring Data,而是 Spring Data Reactive 等,

就算能套用相關的設定與 API,要能獲得 Web Flux 的益處,應用程式中相關的元件,也必須全面檢視,重新設計為非阻斷、基于 Reactive Programming 方式,這或許才是最困難、麻煩的部份,

除了基于 Java 注解的方式,讓熟悉 Web MVC 的開發者容易理解之外,Web Flux 還提供了基于函式式的設計與組態方式,

實際上,在運用 RxJava 2/Reacto r等 Reactive Streams 的實操時,我們也都必須熟悉函式式的思考方式,才能充分掌握,這點在 Web Flux 并不例外,

可以脫離 Servlet 容器了?

Servlet 容器是個舊時代的象征,如果能夠屏蔽 Servlet 容器或相關 API,許多開發者應該都會很開心,可以少一層抽象,不必使用肥肥的 Servlet 容器,當然會是使用 Web Flux 時附帶的優點,然而,如果只是為了屏蔽 Servlet,其實,早就有其他技術選擇存在,

基于 Servlet 一路發展過來的 Web MVC,雖然目前在某些地方可以安插一些函式式的設計,然而,本質上不變的部分在于,在技術堆疊中所隱含的,仍是一個基于同步、阻斷式、命令式的心智模型,如果在這樣的堆疊中,開發者老是因為想要實作非同步、非阻斷、Reactive、函式式而感到不快,Web Flux 也許才會是可考慮的方案,而不單只是用來作為脫離 Servlet 容器,Web MVC 的替代品,

整體而言,Web Flux 還算是新技術,也還有待時間驗證可行性,如果只是為了想用 Web Flux 來取代 Web MVC,或者更小一點的野心,只是想要能脫離 Servlet 容器,最好在采取行動之前,全面檢視一下,確認自身或團隊成員是否準備好接受 Web Flux 的心智模型,或者真的存在著對應的應用場景吧,

 

最近整理一套Java學習資料,包括了Java語法、面向物件、例外、常用類、集合、IO流、多執行緒、網路編程、JUnit、列舉、注解、反射機制、CSS、HTML、JavaScript、JQuery、ajax、Javaweb、MySQL、JavaEE、SSM框架、springboot、演算法、Git、maven、Linux、設計模式、多執行緒與高并發、視頻教程、視頻原始碼、實戰專案、電子書、簡歷模板、高頻面試題等等資源,需要的朋友可以:點擊這里!!暗號博客園!

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/213520.html

標籤:其他

上一篇:Java8 新特性 —— 函式式編程

下一篇:基于Spring Cloud Zookeeper的服務注冊與發現

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more