百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

反应式编程之Spring Web-Flux/Project Reactor

yuyutoo 2025-01-14 18:41 1 浏览 0 评论

介绍

反应式编程代表了我们对应用程序执行模型的看法的改变。在响应式应用程序中,执行不遵循一个请求由一个线程处理的线性模型,而是以事件驱动和非阻塞的方式处理多个请求。

反应式编程提供了一种基于事件的异步流式编程模型,可以处理来自单个或多个客户端的大量并发请求。反应式应用程序通常需要少量线程来垂直扩展,而不是水平扩展。使用响应式编程设计和实现应用程序使应用程序能够最大限度地利用 CPU,从而使应用程序比传统的 Java Web 应用程序具有更高的性能和效率。

反应式编程的特点:

1.异步

2.非阻塞

3. 事件驱动的数据处理方法(事件驱动架构)

术语“反应式”是指围绕对变化做出反应而构建的编程模型,例如对 I/O 事件做出反应的网络组件、对鼠标事件做出反应的 UI 控制器等。从这个意义上说,非阻塞是反应性的,因为我们现在不是被阻塞,而是在操作完成或数据可用时对通知做出反应。(将看到如何发生这种情况的示例)

反应式编程的实现

1. 反应式流(Reactive Streams)

2. RxJava 2.0(最初由Netflix开发,后来开源)

3. Project Reactor(本文)

4.Vert.x

5.其他


Project Reactor的特点:

1. I/O 操作的异步和非阻塞

2. 背压

3.错误处理

4. 处理来自单个或多个客户端的大量并发请求。

5.函数式编程

6. 反应式流是基于推送的,从而提高了性能

7.简单易用的 API (用于 [0|1] 元素) 和(用于 [N] 元素)

I/O 调用非阻塞

反应式和非阻塞式的主要预期好处是能够使用少量固定数量的线程和更少的内存进行扩展。

阻塞可能是浪费,总的来说,有两种方法可以提高程序的性能:

1.并行化以使用更多线程和更多硬件资源。

2.在如何使用当前资源方面寻求更高的效率。

如果仔细观察,一旦程序涉及一些延迟(特别是 I/O,例如数据库请求或网络调用),就会浪费资源,因为线程(可能很多线程)现在处于空闲状态,等待数据。


阻塞请求处理(每个连接一个线程)

非阻塞异步请求处理(事件回调)


事件处理是通过使用:Channel、Buffer 和 Selector 来实现的。

下面列出了一些常用的Channel:

  • FileChannel → 通过文件读写
  • SocketChannel → 通过 TCP 套接字读写
  • ServerSocketChannel →监听客户端的 TCP 连接,为每个 TCP 连接创建新的 SocketChannel
  • DatagramChannel →通过UDP协议读写


Buffer可以看成是一个数据容器,可以用数组来实现。无论是从通道读取还是写入通道,都必须先将数据放入缓冲区。

线程选择器通道

Selector是 Java NIO 中的核心类,它监视和处理在多个注册的 Channel 中发生的感兴趣的 IO 事件。通过这种机制,我们可以只用一个线程来维护多个连接。只有当这些连接之间真正发生了 IO 事件时,才会真正调用 IO 流程逻辑。每次有新连接进来时,不需要启动一个新线程,因此可以显着降低系统负载。

与 Selector 一起,SelectionKey 是另一个重要的类,它代表一个到达的事件。这两个类构成了 NIO 服务器的关键逻辑。

上图显示了在单个线程上运行的 Selector 对多个通道(或连接)的监控。


背压

背压就是下游比上游处理的慢时的一种反馈信号。

Reactive Streams 中的背压概念既优雅又强大。它将允许在响应式应用程序中使用缓慢的消费者,而不会“阻塞”太多信息。可以缓冲未处理的数据。

当消费者在生产者上订阅自己时,它将获得一个订阅。这将启用从数据流的消费者到其生产者的反馈机制。通过它,消费者可以指示他能够处理多少数据事件。

当 Consumer 发出可以处理 5 个数据事件的信号时,Producer 将使用 onNext 方法最多调用 Consumer 5 次。消费完这 5 个事件后,Consumer 会向 Producer 请求额外的事件,直到发生 onComplete 或 onError 调用。


简单易用的 API - Mono(用于 [0|1] 元素)和 Flux(用于 [N] 元素)

Spring WebFlux 围绕 2 个 API 的 Flux 和 Mono 展开 ……

Flux,一个 0-N 项的异步序列

下图显示了如何Flux转换过程:

Mono,异步 0-1 结果

下图显示了如何Mono转换过程:


代码示例:

非反应性代码

@GetMapping("/tweets-blocking")
public List<Tweet> getTweetsBlocking() {
  log.info("Starting BLOCKING Controller!”);
  final String uri = getSlowServiceUri();
  ResponseEntity<List<Tweet>> response = new RestTemplate().exchange(uri, HttpMethod.GET, null, new ParameterizedTypeReference<List<Tweet>>(){});
  List<Tweet> result = response.getBody();
  result.forEach(tweet -> log.info(tweet.toString()));
  log.info("Exiting BLOCKING Controller!");
  return result;
}


反应式代码

@GetMapping(value = "/tweets-non-blocking")
public Flux<Tweet> getTweetsNonBlocking() {
    log.info("Starting NON-BLOCKING Controller!");
    WebClient.create()
        .get()
        .uri(getSlowServiceUri())
        .retrieve()
        .bodyToFlux(Tweet.class)
        .subscribe(tweet -> log.info(tweet.toString()));
    log.info("Exiting NON-BLOCKING Controller!");
    return tweetFlux;
}

两者不同的一点是:反应式中直到订阅的时候(subscribes)才会处理,即延迟执行/懒式执行。

总结

Spring Boot 反应式(Spring Webflux)与非反应式堆栈的比较


有什么优点和缺点?

优点

  • 代码更简洁,更简洁
  • 更容易阅读(一旦你掌握了它)
  • 更容易扩展(管道任何操作)
  • 更好的错误处理
  • 事件驱动的启发 -> 与流(Kafka、RabbitMQ 等)配合得很好
  • 背压(客户端可以控制流量)
  • 非阻塞 I/O 操作(这意味着更好的响应时间)

缺点

· 大部分时间存储数据流需要更多内存(因为它是基于一段时间的流)

  • 一开始可能会觉得学习非常规(需要一切都是流)

· 与传统java风格相比不同的编程风格

· 一切都必须围绕 Mono/Flux

什么时候使用?

实现高吞吐量

系统中的大量数据流

什么时候不使用

当系统内没有太多数据需要处理时

什么是反应式系统?

反应式系统与反应式编程不同。反应式编程用于代码级别,而反应式系统处理架构。

·响应能力:对用户可用,并且无论发生什么(过载、故障等),都准备好响应他们。

·弹性:保持不受故障、中断和极高负载的影响。

·弹性:有效利用资源并平衡机器性能——垂直扩展或缩减——或轻松调节涉及的机器数量——水平扩展或缩减——取决于负载。

·消息驱动特性:通过向可寻址的接收者发送不可变消息来实现完全无阻塞的通信。

官网:https://projectreactor.io/docs/core/release/reference/#getting-started-introducing-reactor

相关推荐

.NET Core 中推荐使用的10大优秀库,你用到过几个?

概述:Microsoft的.NETCore生态系统中的中间件已经发生了重大变化,包括无缝集成到应用程序管道中的内置和第三方组件,协调客户端和服务器之间的数据流。它通过身份验证、日志记录和路由等...

机器学习中英文对照表

10-1LossFunction0-1损失函数2Accept-RejectSamplingMethod接受-拒绝抽样法/接受-拒绝采样法3AccumulatedErrorBa...

反应式编程之Spring Web-Flux/Project Reactor

介绍反应式编程代表了我们对应用程序执行模型的看法的改变。在响应式应用程序中,执行不遵循一个请求由一个线程处理的线性模型,而是以事件驱动和非阻塞的方式处理多个请求。...

Spider详解

简介Spider的功能主要使用于大型的应用系统测试,它能在很短的时间内帮助我们快速地对一个应用程序的内容、功能、系统的结构和分布情况进行了解。Control右键进行爬取数据使用spider功能。在Sp...

WebUI 如何高效进行测试

1.选择合适的浏览器驱动ChromeDriver:对于大多数情况,推荐使用ChromeDriver,因为它与Chrome浏览器的兼容性好,并且性能较好。...

《成为Rust专家》五、单元测试 (2)

6.3测试框架Rust的单元测试不包括其他单元测试框架中可能找到的辅助函数、夹具、测试框架或参数化测试功能。对于这些功能,你需要自己编写代码或者尝试一些库。对于基本的参数化测试,parameteri...

JUnit5学习之一:基本操作

欢迎访问我的GitHubhttps://github.com/zq2599/blog_demos内容:所有原创文章分类和汇总,及配套源码,涉及Java、Docker、Kubernetes、DevOPS...

511基于C# Thread类的大漠多线程模板游戏实战

如果你的游戏检测易语言,或者,客户反馈你的脚本被频繁报毒,加入黑名单,那么我们选择微软的C#来写一个大漠的多线程模板是最好的选择。...

如何深度理解mybatis?

深度自定义mybatis回顾mybatis的操作的核心步骤...

.NET 6 多线程的几种打开方式

前言多线程无处不在,平常的开发过程中,应该算是最常用的基础技术之一了。以下通过Thread、ThreadPool、再到Task、Parallel、线程锁、线程取消等方面,一步步进行演示多线程的一些基础...

C# 多 线 程。

一、基本概念1、进程...

C#多线程

1.概念进程,线程,应用程序的定义网上有很多资料,但是有些抽象。通俗的来讲,进程就是一旦一个应用程序开始运行,那么这个应用程序就会存在一个属于这个应用程序的进程。线程就是进程中的基本执行单元,每个进...

多线程在C# (.NET) 中的应用

在实际项目应用中我们难免会用到多线程、多进程编程方式,C#中的多线程允许你在同一时间内执行多个线程,每个线程都可以独立地执行不同的任务或者处理不同的部分。这可以帮助提高应用程序的响应性和性能。通过这...

如何使?C#创建?个线程?

在C#中,可以通过多种方式创建和启动一个线程。以下是常用的方式及其具体实现。1.使用Thread类创建线程...

在C#中,如何创建并启动?个新的线程?请举例说明

在C#中,可以使用System.Threading.Thread类创建并启动一个新的线程。以下是创建和启动线程的方式以及示例代码:创建并启动线程的步骤...

取消回复欢迎 发表评论: