【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&#…...

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式
一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明:假设每台服务器已…...
Python|GIF 解析与构建(5):手搓截屏和帧率控制
目录 Python|GIF 解析与构建(5):手搓截屏和帧率控制 一、引言 二、技术实现:手搓截屏模块 2.1 核心原理 2.2 代码解析:ScreenshotData类 2.2.1 截图函数:capture_screen 三、技术实现&…...

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?
编辑:陈萍萍的公主一点人工一点智能 未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战,在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…...

微信小程序之bind和catch
这两个呢,都是绑定事件用的,具体使用有些小区别。 官方文档: 事件冒泡处理不同 bind:绑定的事件会向上冒泡,即触发当前组件的事件后,还会继续触发父组件的相同事件。例如,有一个子视图绑定了b…...
vue3 字体颜色设置的多种方式
在Vue 3中设置字体颜色可以通过多种方式实现,这取决于你是想在组件内部直接设置,还是在CSS/SCSS/LESS等样式文件中定义。以下是几种常见的方法: 1. 内联样式 你可以直接在模板中使用style绑定来设置字体颜色。 <template><div :s…...
在四层代理中还原真实客户端ngx_stream_realip_module
一、模块原理与价值 PROXY Protocol 回溯 第三方负载均衡(如 HAProxy、AWS NLB、阿里 SLB)发起上游连接时,将真实客户端 IP/Port 写入 PROXY Protocol v1/v2 头。Stream 层接收到头部后,ngx_stream_realip_module 从中提取原始信息…...

前端开发面试题总结-JavaScript篇(一)
文章目录 JavaScript高频问答一、作用域与闭包1.什么是闭包(Closure)?闭包有什么应用场景和潜在问题?2.解释 JavaScript 的作用域链(Scope Chain) 二、原型与继承3.原型链是什么?如何实现继承&a…...

JUC笔记(上)-复习 涉及死锁 volatile synchronized CAS 原子操作
一、上下文切换 即使单核CPU也可以进行多线程执行代码,CPU会给每个线程分配CPU时间片来实现这个机制。时间片非常短,所以CPU会不断地切换线程执行,从而让我们感觉多个线程是同时执行的。时间片一般是十几毫秒(ms)。通过时间片分配算法执行。…...

用docker来安装部署freeswitch记录
今天刚才测试一个callcenter的项目,所以尝试安装freeswitch 1、使用轩辕镜像 - 中国开发者首选的专业 Docker 镜像加速服务平台 编辑下面/etc/docker/daemon.json文件为 {"registry-mirrors": ["https://docker.xuanyuan.me"] }同时可以进入轩…...
MySQL用户和授权
开放MySQL白名单 可以通过iptables-save命令确认对应客户端ip是否可以访问MySQL服务: test: # iptables-save | grep 3306 -A mp_srv_whitelist -s 172.16.14.102/32 -p tcp -m tcp --dport 3306 -j ACCEPT -A mp_srv_whitelist -s 172.16.4.16/32 -p tcp -m tcp -…...