【Spring Boot】响应式编程
响应式编程
- 1.WebFlux
- 2.比较 MVC 和 WebFlux
- 2.1 工作方式
- 2.2 Spring MVC 与 Spring WebFlux 的区别
- 2.3 使用 WebFlux 的好处
- 3.Mono 和 Flux
- 3.1 Mono 和 Flux 是什么
- 3.2 Mono 和 Flux 的区别
- 4.开发 WebFlux 的流程
- 4.1 注解式开发流程
- 4.2 响应式开发流程
- 5.用注解式开发实现 Hello World
- 5.1 配置 WebFlux 依赖
- 5.2 编写控制器
- 6.用响应式开发方式开发 WebFlux
- 6.1 编写处理器类 Handler
- 6.2 编写路由器类 Router
1.WebFlux
WebFlux 是从 Spring Framework 5.0 开始引入响应式 Web 框架的。与 Spring MVC 不同,WebFlux 不需要 ServletAPI,在完全异步且无阻塞,并通过 Reactor 项目实现 Reactive Streams 规范。
WebFlux 可以在资源有限的情况下提高 系统的吞吐量和伸缩性(不是提高性能)。这意味着在资源相同的情况下,WebFlux 可以处理更多的请求(不是业务)。
WebFlux 除支持 RESTful Web 服务外,还可以用于提供动态 HTML 内容。
2.比较 MVC 和 WebFlux
Spring MVC 采用命令式编程的方式,代码被一句一句地执行,便于开发者理解与调试代码。WebFlux 则是基于异步响应式编程。
2.1 工作方式
- MVC:主线程接收到请求(
request
)→ 准备数据 → 返回数据。整个过程是单线程阻塞的,用户会感觉等待时间长是因为,在结果处理好之后才返回数据给浏览器。因此,如果请求很多,则吞吐量就上不去。 - WebFlux:主线程接收到请求 → 立刻返回数据与的数的组合(
Mono
或Flux
,不是结果) → 开启一个新 Work 线程去做实际的数据准备工作,进行真正的业务操作 → Work 线程完成工作 → 返给用户真实数据(结果)。这种方式给人的感觉是响应时间很短,因为返回的是不变的常数,它不随用户数量的增加而变化。
2.2 Spring MVC 与 Spring WebFlux 的区别
对比项 | | |
---|---|---|
地址(路由)映射 | @Controller、@RequestMapping 等标准的 Spring MVC 注解 | (1)Router Functions,提供一套函数式风格的 API,用于创建 Router、Handler 和 Filter (2)@Controller、@RequestMapping 等标准的 Spring MVC 注解 |
数据流 | Servlet API | Reactive Streams:一种支持背压(backpressure )的异步数据流处理标准,主流实现有 RxJava 和 Reactor 。Spring WebFlux 默认集成的是 Reactor |
容器 | Tomcat、Jetty、Undertow | Netty、Tomcat、Jetty、Undertow |
I/O 模型 | 同步阻塞的 I/O 模型 | 异步非阻塞的 I/O 模型 |
吞吐性能 | 低 | 高 |
业务处理性能 | 一样 | 一样 |
支持数据库 | NoSQL、SQL | 支持 NoSQL,不支持 MySQL 等关系型数据库 |
请求和响应 | HttpServletRequest 和 HttpServletResponse | ServerRequest 和 ServerResponse |
2.3 使用 WebFlux 的好处
下面以餐厅 “叫号” 来比喻阻塞式开发与 WebFlux。
假设 “海底捞” 没有叫号机(前台服务员),店里有 200 个餐台供客人进餐,如果此时来了 201 个客人,那么最后一个客人就直接被拒绝服务了。
而现在有叫号机,来了 200 个客人正在用餐,后面再来 100 个客人,叫号机马上给后面的 100 个客人每人一个排队号。这样服务就不阻塞了,每个人都立马得到反馈。来再多的人也能立马给排号,但是进餐依然是阻塞的。
回到程序。我们假设,服务器最大线程资源数为 200 个,当前遇到 200个非常耗时的请求,如果再来 1 个请求时,阻塞式程序就已经处理不了(拒绝服务)了。
而对于 WebFlux,则可以做到立即响应(告诉用户等着),然后将收到的请求转发给 Work 线程去处理。WebFlux 只会对 Work 线程形成阻塞,如果再来请求也可以处理。其主要应用场景是在业务处理较耗时的场景中减少服务器资源的占用,提高并发处理速度。
对 WebFlux 的一个简单的理解就是:你来了,我立马应答你,但是服务需要等待;而不是你来了没人理你,咨询服务半天也回复不了。
结论:MVC 能满足的场景,就不需要改用 WebFlux。WebFlux 和 MVC 可以混合使用。如果开发 I/O 密集型服务,则可以选择用 WebFlux 实现。
如果在
pom.xml
文件中同时引用了spring-boot-starter-web
和spring-boot-starter-webflux
依赖,则优先会使用spring-boot-starter-web
。这时,控制台输出的启动日志会提示 “Tomcat started on port(s): 8080 (http) with context path”,而使用 WebFlux 会提示 “Netty started onport(s): 8080”。
3.Mono 和 Flux
3.1 Mono 和 Flux 是什么
Mono 和 Flux 是 Reactor 中的两个基本概念。
- Mono 和 Flux 属于 事件发布者,为消费者提供订阅接口。当有事件发生时,Mono 或 Flux 会回调消费者的相应方法,然后通知消费者相应的事件。这也是响应式编程模型。
- Mono 和 Flux 用于处理异步数据流,它不像 MVC 中那样直接返回 String/List,而是将异步数据流包装成 Mono 或 Flux 对象。
3.2 Mono 和 Flux 的区别
-
Flux 可以发送很多
item
,并且这些item
可以经过若干算子(operators
)后才被订阅。Mono 只能发送一个item
。 -
Mono 主要用于返回单个数据。Flux 用于返回多个数据。如果要根据
id
查询某个 User 对象,则返回的肯定是单个 User,那么需要将其包装成Mono<User>
。若需要获取所有 User(这是一个集合),则需要将这个集合包装成Flux<User>
。这里的单个数据并不是指一个数据,而是指封装好的一个对象。多个数据就是多个对象。 -
Mono 表示包含 0 或 1 个元素的异步序列。在该序列中可以包含 3 种不同类型的消息通知:正常的包含元素的消息、序列结束的消息、序列出错的消息。当消息通知(正常的包含元素的消息、序列结束的消息、序列出错的消息)产生时,订阅者中有对应的方法
onNext()
、onComplete()
、onError()
被调用。 -
Flux 表示的是包含 0 到 N 个元素的异步序列,在该序列中可以包含与 Mono 相同的 3 种类型的消息通知。
-
Flux 和 Mono 之间可以进行转换。对一个 Flux 序列进行计数操作时,得到的结果是
Mono<Long>
对象。把多个 Mono 序列合并在一起,得到的是一个 Flux 对象。
4.开发 WebFlux 的流程
4.1 注解式开发流程
WebFlux 是响应式框架,其中使用的 注解式开发方式 只是 Spring 团队为了更好地迁移而提的。和 MVC 开发模式一样,地址映射也是通过 @RequestMapping 提供的,用 @Controller 或 @RestController 来代替 Handler 类。
4.2 响应式开发流程
(1)创建 Handler 类。这里的 Handler 类相当于 Spring MVC 的 Controller 层中的方法体。在响应式编程中,请求和响应不再是 HttpServletRequest 和 HttpServletResponse,而是变成了 ServerRequest 和 ServerResponse。
(2)配置 RouterFunction。RouterFunction 和注解 @RequestMapping 相似,都用于提供 URL 路径。RouterFunction 的格式也是固定的,第 1 个参数代表 路径,第 2 个参数代表 方法,合起来代表将 URL 映射到方法。
5.用注解式开发实现 Hello World
5.1 配置 WebFlux 依赖
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-webflux</artifactId>
</dependency>
5.2 编写控制器
用注解式开发 WebFlux 应用程序与 MVC 的开发方式是一样的。通过注解 @RestController 标注控制器类,通过注解 @GetMapping 指定映射路径。
package com.example.demo;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
import reactor.core.publisher.Mono;@RestController
public class HelloWorldController {@GetMapping("/helloworld")public Mono<String> helloworld(){return Mono.just("This is WebFlux demo");}
}
启动工程后,控制台中输出如下:
可以看到,WebFlux 默认使用的是 Netty 服务器,而不是 MVC 模式下的 Tomcat 服务器。
6.用响应式开发方式开发 WebFlux
6.1 编写处理器类 Handler
Handler 相当于 MVC 中的 Controller。用于提供实现功能的方法。
package com.example.demo;import org.springframework.http.MediaType;
import org.springframework.stereotype.Component;
import org.springframework.web.reactive.function.server.ServerRequest;
import org.springframework.web.reactive.function.server.ServerResponse;
import reactor.core.publisher.Mono;@Component
public class HelloWorldHandler {public Mono<ServerResponse> sayHelloWorld(ServerRequest serverRequest) {return ServerResponse.ok().contentType(MediaType.TEXT_PLAIN).body(Mono.just("This is WebFlux demo"), String.class);}
}
6.2 编写路由器类 Router
package com.example.demo;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.reactive.function.server.RouterFunction;
import org.springframework.web.reactive.function.server.ServerResponse;import static org.springframework.web.reactive.function.server.RequestPredicates.GET;
import static org.springframework.web.reactive.function.server.RouterFunctions.route;@Configuration
public class Router {@Autowiredprivate HelloWorldHandler helloWorldHandler;@Beanpublic RouterFunction<ServerResponse> getString(){return route(GET("/helloworld"),req->helloWorldHandler.sayHelloWorld(req));}
}
上述代码中,通过 return route(GET("/helloworld"),req->helloWorldHandler.sayHelloWorld(req));
来指定路由,包含 HTTP 方法和对应的功能方法。
相关文章:

【Spring Boot】响应式编程
响应式编程 1.WebFlux2.比较 MVC 和 WebFlux2.1 工作方式2.2 Spring MVC 与 Spring WebFlux 的区别2.3 使用 WebFlux 的好处 3.Mono 和 Flux3.1 Mono 和 Flux 是什么3.2 Mono 和 Flux 的区别 4.开发 WebFlux 的流程4.1 注解式开发流程4.2 响应式开发流程 5.用注解式开发实现 He…...

【C++练级之路】【Lv.21】C++11——列表初始化和声明
快乐的流畅:个人主页 个人专栏:《算法神殿》《数据结构世界》《进击的C》 远方有一堆篝火,在为久候之人燃烧! 文章目录 引言一、列表初始化1.1 内置类型1.2 结构体或类1.3 容器 二、声明2.1 auto2.2 decltype2.3 nullptr 三、STL的…...

输入一串字符串,前中后都有*号,去掉字符串中间和后面的*号,保留前面的*号和字母
#include <stdio.h> void fun(char* a) {//***df**fr*fg***int i 0, j 0,n0,m0;char* p;p a;while (p[i] ! \0){i;//i是一共的字符的个数}printf("%d\n",i);while (a[n] *){n;//计算字母前的*的个数}printf("%d\n", n);m n;for (j n; j < …...

【机器学习与大模型】驱动下的应用图像识别与处理
摘要: 本文深入探讨了机器学习在图像识别与处理领域的应用,特别是在大模型的推动下所取得的巨大进展。详细阐述了图像识别与处理的基本原理、关键技术,以及机器学习算法和大模型如何提升其性能和准确性。通过实际案例分析了其在多个领域的广泛…...

24李林跌落神坛,880还刷吗?还是换1000、900、660?
“李林今年跌落神坛了!” “全是固定题型没新题,结果今年考的全是新题。” 880是“老真题的神”, 遇到24年,冷门考点多,计算量又大,就不灵了。 但“老真题”,还是得刷。就像往年真题是要刷的…...
数据库漫谈-sybase
sybase就是“system”加“database”,代表着信息系统的底层。Sybase公司很早就推出了关系数据库产品(1987年5月推出的Sybase SQLServer1.0)。Sybase也是第一个提出Client/Server 体系结构的思想,并率先在Sybase SQLServer 中实现。…...

Springboot开发 -- Postman 调试类型详解
引言 在 Spring Boot 应用开发过程中,接口测试是必不可少的一环。Postman 作为一款强大的 API 开发和测试工具,可以帮助开发者轻松构建、测试和管理 HTTP 请求。本文将为大家介绍如何在 Spring Boot 开发中使用 Postman 进行接口测试。 一、准备工作 安…...
Windows 后台启动jar并且输出日志到特定日志
Windows 后台启动jar并且输出日志到特定日志 javaw -Dfile.encodingutf-8 -jar xxx.jar >log.log 2>&1 &日志输出以年月日格式显示 javaw -Dfile.encodingutf-8 -jar xxx.jar >log_%DATE:~0,4%-%DATE:~5,2%-%DATE:~8,2%_%TIME:~0,2%-%TIME:~3,2%-%TIME:~6,2…...

垃圾回收机制及算法
文章目录 概要对象存活判断引用计数算法可达性分析算法对象是否存活各种引用 垃圾收集算法分代收集理论复制算法标记清除算法标记-整理算法 概要 垃圾收集(Garbage Collection, 下文简称GC),其优缺点如下: 优点&#…...
蓝桥杯-暴力搜索BFS+DFS
九九乘法表挂毯 问题描述: 在一个古老的城堡里,一位名为 Alex 的少年发现了一幅巨大的九九乘法表挂毯。挂毯被划分成了9x9的方格,每个方格上写着相应的乘积。Alex 想象自己站在数值为1的方格上,他的目标是到达数值为 81 的方格。…...
巧用count与count()
在C#中,talentInnoPfChains.Count() 和 talentInnoPfChains.Count 的性能差异主要取决于 talentInnoPfChains 的类型。这里有两种可能的情况: 如果 talentInnoPfChains 是一个实现了 ICollection<T> 接口的集合(如 List<T>, Hash…...
MongoDB 覆盖索引查询:提升性能的完整指南
MongoDB 覆盖索引查询是一种优化数据库查询性能的技术,它通过创建适当的索引,使查询可以直接从索引中获取所需的数据,而无需访问实际的文档数据。这种方式可以减少磁盘 I/O 和内存消耗,提高查询性能。 基本语法 在 MongoDB 中&a…...
ECMAScript详解
ECMAScript(简称ES)是一种由Ecma国际(前身为欧洲计算机制造商协会,European Computer Manufacturers Association)通过ECMA-262标准化的脚本程序设计语言。以下是对ECMAScript的详细说明: 1. 定义与起源 …...

如何在Windows 10上对硬盘进行碎片整理?这里提供步骤
随着时间的推移,由于文件系统中的碎片,硬盘驱动器可能会开始以较低的效率运行。为了加快驱动器的速度,你可以使用内置工具在Windows 10中对其进行碎片整理和优化。方法如下。 什么是碎片整理 随着时间的推移,组成文件的数据块&a…...

科学高效备考AMC8和AMC10竞赛,吃透2000-2024年1850道真题和解析
多做真题,吃透真题和背后的知识点是备考AMC8、AMC10有效的方法之一,通过做真题,可以帮助孩子找到真实竞赛的感觉,而且更加贴近比赛的内容,可以通过真题查漏补缺,更有针对性的补齐知识的短板。 今天我们继续…...

SQL——SELECT相关的题目
目录 197、上升的温度 577、员工奖金 586、订单最多的客户 596、超过5名学生的课 610、判断三角形 620、有趣的电影 181、超过经理收入的员工 1179、重新格式化部门表(行转列) 1280、学生参加各科测试的次数 1068、产品销售分析I 1075、项目员工I …...

etcd集群部署
1.etcd介绍 1.1 什么是etcd etcd的官方定义如下: A distributed, reliable key-value store for the most critical data of distributed systemetcd是一个Go语言编写的分布式、高可用的一致性键值存储系统,用于提供可靠的分布式键值(key value)存储、配置共享和服务发现等…...

VBA_MF系列技术资料1-615
MF系列VBA技术资料1-615 为了让广大学员在VBA编程中有切实可行的思路及有效的提高自己的编程技巧,我参考大量的资料,并结合自己的经验总结了这份MF系列VBA技术综合资料,而且开放源码(MF04除外),其中MF01-0…...
常用激活函数学习
常用激活函数及其应用 ReLU (Rectified Linear Unit) 公式: f ( x ) max ( 0 , x ) f(x) \max(0, x) f(x)max(0,x)理解: 当输入值为正时,输出等于输入值;否则输出为0。ReLU函数简单且计算效率高,能有效缓解梯度消失问题,促进…...

html中被忽略的简单标签
1: alt的作用是在图片不能显示时的提示信息 <img src"https://img.xunfei.cn/mall/dev/ifly-mall-vip- service/business/vip/common/202404071019208761.jp" alt"提示信息" width"100px" height"100px" /> 2&#…...

XCTF-web-easyupload
试了试php,php7,pht,phtml等,都没有用 尝试.user.ini 抓包修改将.user.ini修改为jpg图片 在上传一个123.jpg 用蚁剑连接,得到flag...
【根据当天日期输出明天的日期(需对闰年做判定)。】2022-5-15
缘由根据当天日期输出明天的日期(需对闰年做判定)。日期类型结构体如下: struct data{ int year; int month; int day;};-编程语言-CSDN问答 struct mdata{ int year; int month; int day; }mdata; int 天数(int year, int month) {switch (month){case 1: case 3:…...

《从零掌握MIPI CSI-2: 协议精解与FPGA摄像头开发实战》-- CSI-2 协议详细解析 (一)
CSI-2 协议详细解析 (一) 1. CSI-2层定义(CSI-2 Layer Definitions) 分层结构 :CSI-2协议分为6层: 物理层(PHY Layer) : 定义电气特性、时钟机制和传输介质(导线&#…...

【Redis技术进阶之路】「原理分析系列开篇」分析客户端和服务端网络诵信交互实现(服务端执行命令请求的过程 - 初始化服务器)
服务端执行命令请求的过程 【专栏简介】【技术大纲】【专栏目标】【目标人群】1. Redis爱好者与社区成员2. 后端开发和系统架构师3. 计算机专业的本科生及研究生 初始化服务器1. 初始化服务器状态结构初始化RedisServer变量 2. 加载相关系统配置和用户配置参数定制化配置参数案…...

ESP32读取DHT11温湿度数据
芯片:ESP32 环境:Arduino 一、安装DHT11传感器库 红框的库,别安装错了 二、代码 注意,DATA口要连接在D15上 #include "DHT.h" // 包含DHT库#define DHTPIN 15 // 定义DHT11数据引脚连接到ESP32的GPIO15 #define D…...
工程地质软件市场:发展现状、趋势与策略建议
一、引言 在工程建设领域,准确把握地质条件是确保项目顺利推进和安全运营的关键。工程地质软件作为处理、分析、模拟和展示工程地质数据的重要工具,正发挥着日益重要的作用。它凭借强大的数据处理能力、三维建模功能、空间分析工具和可视化展示手段&…...

C++ 求圆面积的程序(Program to find area of a circle)
给定半径r,求圆的面积。圆的面积应精确到小数点后5位。 例子: 输入:r 5 输出:78.53982 解释:由于面积 PI * r * r 3.14159265358979323846 * 5 * 5 78.53982,因为我们只保留小数点后 5 位数字。 输…...

ABAP设计模式之---“简单设计原则(Simple Design)”
“Simple Design”(简单设计)是软件开发中的一个重要理念,倡导以最简单的方式实现软件功能,以确保代码清晰易懂、易维护,并在项目需求变化时能够快速适应。 其核心目标是避免复杂和过度设计,遵循“让事情保…...

基于Java+MySQL实现(GUI)客户管理系统
客户资料管理系统的设计与实现 第一章 需求分析 1.1 需求总体介绍 本项目为了方便维护客户信息为了方便维护客户信息,对客户进行统一管理,可以把所有客户信息录入系统,进行维护和统计功能。可通过文件的方式保存相关录入数据,对…...

GitFlow 工作模式(详解)
今天再学项目的过程中遇到使用gitflow模式管理代码,因此进行学习并且发布关于gitflow的一些思考 Git与GitFlow模式 我们在写代码的时候通常会进行网上保存,无论是github还是gittee,都是一种基于git去保存代码的形式,这样保存代码…...