再见了SpringMVC,这个框架有点厉害,甚至干掉了Servlet!

前言

对 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 下手,毕竟也是个方法。ide

编写程序时,就算没用到 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 的实做品。spa

既然是实现了 Reactive Streams 规范,开发者必然会想到的是 RxJava/RxJava 2,或者是 Java 9 的 Flow API。这也意谓着,在能使用 Web Flux 以前,开发者必须对于 Reactive Programming 典范,有所认识。设计

开发者这时有疑问了,Spring 为什么不直接基于 RxJava 2,而是打造专属的 Reactive Streams 项目呢?orm

就技术而言,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故事程序员,多谢各位啦!

相关文章
相关标签/搜索