首页 > 其他分享 >Spring WebFlux简介

Spring WebFlux简介

时间:2023-04-30 17:01:33浏览次数:48  
标签:web Spring 编程 WebFlux 阻塞 API 简介

一、为什么需要web Flux

部分原因是需要一个无阻塞的web堆栈来处理少量线程的并发性,并用更少的硬件资源进行扩展。Servlet 3.1确实为非阻塞I/O提供了一个API。然而,使用它会偏离Servlet API的其余部分,在那里交互是同步的(Filter,Servlet)或阻塞的(getParameter,getPart)。这就是一个新的通用API作为任何非阻塞运行时的基础的动机。这一点很重要,因为服务器(如Netty)在异步、非阻塞空间中已经建立起来。
 
答案的另一部分是函数编程。正如在Java 5中添加注释创造了机会(如带注释的REST控制器或单元测试)一样,在Java 8中添加lambda表达式也为Java中的函数API创造了机会。这对于允许异步逻辑的声明性组合的非阻塞应用程序和延续式API(CompletableFuture和ReactiveX推广了这一点)来说是一个福音。在编程模型级别,Java8使SpringWebFlux能够在注释控制器的同时提供功能性的web端点。

二、怎么定义Reactive

术语“反应式”指的是围绕对变化做出反应而建立的编程模型 — 对I/O事件做出反应的网络组件、对鼠标事件做出反应,以及其他。从这个意义上说,非阻塞是被动的,因为我们现在不是被阻塞,而是在操作完成或数据可用时对通知做出反应。命令式编程主要采用顺序,分支和循环三种主要结构运行程序。比如运行:

int a = 10;
int b = 20;
int c = a + b;
System.out.println(c);

运行时输出30,当改变a为20时c还是30。在 Excel 里,C 单元格上设置函数 Sum(A+B),当你改变单元格 A 或者单元格 B 的数值时,单元格 C 的值同时也会发生变化。这种行为就是 Reactive。在 Java 9 Flow 中,按相同的思路实现上述处理流程,当初始变量的值变化,最后结果的值也同步发生变化,这就是响应式编程。这相当于声明了一个公式,输出值会随着输入值而同步变化。

  SubmissionPublisher<Integer> publisher = new SubmissionPublisher<>();
  publisher.subscribe(new Flow.Subscriber<Integer>() {
      private Integer sum = 0;
      Flow.Subscription subscription = null;

      @Override
      public void onSubscribe(Flow.Subscription subscription) {

          this.subscription = subscription;
          subscription.request(1);
      }

      @Override
      public void onNext(Integer item) {
          subscription.request(1);
          sum += item;

      }

      @Override
      public void one rror(Throwable throwable) {

      }

      @Override
      public void onComplete() {
          System.out.println(sum);
      }

  });

  Arrays.asList(3, 4).stream().forEach(publisher::submit);
  publisher.close();

 
还有另一个重要的机制,Spring团队将其与“反应性”联系在一起,那就是非阻塞性背压。在同步命令式代码中,阻塞调用是一种自然形式的背压,迫使调用方等待。在非阻塞代码中,控制事件的速率变得很重要,这样快速生产者就不会淹没其目的地。在命令式编程中,当消费者速率赶不上生产者速率时,消费者要么将多出来的元素缓存,要么丢弃。在JAVA 8 Stream中,当执行到终端操作时才会拉取元素,因此Stream只能执行一次。但是Reactive是消费者主动拉取生产者的元素,消费者的消费速率是多少就可以控制从生产者拉取多少。这就是非阻塞性背压。
 
Reactive Streams是一个小规范(也在Java 9中采用),它定义了具有背压的异步组件之间的交互。例如,数据存储库(充当发布服务器)可以生成HTTP服务器(充当订阅服务器)可以写入响应的数据。Reactive Streams的主要目的是让订阅者控制发布者生成数据的速度或速度。在 Java 中,有 4 个 Reactive Streams API,在 JUC 的 Flow 类中可以看到:
 

  • Publisher 即事件的发生源,它只有一个 subscribe 方法。其中的 Subscriber 就是订阅消息的对象。
  • Subscriber 作为订阅者,有四个方法。onSubscribe 会在每次接收消息时调用,得到的数据都会经过 onNext 方法。onError 方法会在出现问题时调用,Throwable 即是出现的错误消息。在结束时调用 onComplete 方法。
  • Subscription 接口用来描述每个订阅的消息。request 方法用来向上游索要指定个数的消息,cancel 方法用于取消上游的数据推送,不再接受消息。
  • Processor 接口继承了 Subscriber 和 Publisher,它既是消息的发生者也是消息的订阅者。这是发生者和订阅者间的过渡桥梁,负责一些中间转换的处理

三、编程模型

spring-web模块包含spring WebFlux基础的反应式基础,包括HTTP抽象、支持的服务器的反应式流适配器、编解码器,以及与Servlet API类似但具有非阻塞契约的核心WebHandler API。

在此基础上,SpringWebFlux提供了两种编程模型的选择:

  • 带注解的控制器:与Spring MVC一致,并基于来自Spring web模块的相同注释。Spring MVC和WebFlux控制器都支持反应式(Reactor和RxJava)返回类型,因此,很难将它们区分开来。一个显著的区别是WebFlux还支持反应式@RequestBody参数。
  • 函数端点:基于Lambda的轻量级功能编程模型。您可以将其视为应用程序可以用来路由和处理请求的一个小型库或一组实用程序。与带注释的控制器的最大区别在于,应用程序负责从开始到结束的请求处理,而不是通过注释声明意图并被调用。

四、如何选择

选择Spring MVC还是选择Spring WebFlux?下图显示了两者之间的关系,它们有什么共同点,以及各自唯一支持什么:

 
建议如下:

  • 如果有一个运行良好的SpringMVC应用程序,那么就没有必要进行更改。命令式编程是编写、理解和调试代码的最简单方法。可以最大限度地选择库,因为从历史上看,大多数库都是阻塞的。
  • 如果想选择非阻塞web堆栈,那么Spring WebFlux提供了与该领域其他执行模型相同的执行模型优势,还提供了服务器选择(Netty、Tomcat、Jetty、Undertow和Servlet 3.1+容器)、编程模型选择(带注解的控制器和功能性web端点)以及反应库选择(Reactor、RxJava或其他)。
  • 如果对用于Java 8 lambdas或Kotlin的轻量级、功能性web框架感兴趣,可以使用Spring WebFlux功能性web端点。对于要求不那么复杂的小型应用程序或微服务来说,这也是一个不错的选择,它们可以从更大的透明度和控制中受益。
  • 在微服务架构中,可以将应用程序与Spring MVC或Spring WebFlux控制器或Spring WebFlux功能端点混合使用。在两个框架中都支持相同的基于注释的编程模型,可以更容易地重用知识,同时为正确的工作选择正确的工具。
  • 评估应用程序的一种简单方法是检查其依赖关系。如果有阻塞持久性API(JPA、JDBC)或网络API可供使用,那么Spring MVC至少是通用体系结构的最佳选择。Reactor和RxJava在单独的线程上执行阻塞调用在技术上是可行的,但不会充分利用非阻塞的web堆栈。
  • 如果有一个可以调用远程服务的SpringMVC应用程序,请尝试响应式WebClient。可以直接从Spring MVC控制器方法返回反应类型(Reactor、RxJava或其他)。每次调用的延迟或调用之间的相互依赖性越大,好处就越显著。Spring MVC控制器也可以调用其他响应组件。

 
 
 

参考自:
一文弄懂 Spring WebFlux 的来龙去脉

标签:web,Spring,编程,WebFlux,阻塞,API,简介
From: https://www.cnblogs.com/shigongp/p/17351011.html

相关文章

  • SpringSecurity过滤器之SessionManagementFilter
    SessionManagementFilter检测用户自请求开始以来是否已通过身份验证,如果已通过,则调用SessionAuthenticationStrategy以执行任何与会话相关的活动,例如激活会话固定保护机制或检查多个并发登录。配置如下:@ConfigurationpublicclassMySecurityConfigextendsWebSecurityConfigur......
  • 【驱动】以太网扫盲(二)phy寄存器简介
    PHY寄存器的地址空间为5位,从0到31最多可以定义32个寄存器(随着芯片功能不断增加,很多PHY芯片采用分页技术来扩展地址空间以定义更多的寄存器),IEEE802.3定义了地址为0-15这16个寄存器的功能,地址16-31的寄存器留给芯片制造商自由定义,如下表所示。官方介绍请参考......
  • 一天吃透SpringBoot面试八股文
    Springboot的优点内置servlet容器,不需要在服务器部署tomcat。只需要将项目打成jar包,使用java-jarxxx.jar一键式启动项目SpringBoot提供了starter,把常用库聚合在一起,简化复杂的环境配置,快速搭建spring应用环境可以快速创建独立运行的spring项目,集成主流框架准生产环境的......
  • 06 Spring集合注入&读取properties文件
    文章目录1Spring注入集合1.1在目标类中添加接收集合的的变量并提供setter方法1.2在配置文件中进行配置2读取properties文件2.1新开一个命名空间2.2在resources目录下新建.properties配置文件2.3在spring配置文件applicationContext.xml中引用配置文件信息2.4配置文件需要......
  • 03 Spring生命周期
    文章目录1方式1标签属性配置1.1在目标类中添加初始化、销毁方法[init(),destroy()]1.2在spring配置文件中对应的bean中添加属性2通过spring接口方式在目标类中实现接口,InitializingBean和DisposableBean两接口3关闭容器的两种操作3.1使用close()方法3.2使用钩子函数:register......
  • Spring 的 IoC(控制反转)
    IoC是InversionofControl的简写,译为“控制反转”,它不是一门技术,而是一种设计思想,也是一个概念,同时是一个重要的面向对象编程法则,能够指导我们如何设计出松耦合、更优良的程序。在这里说IoC之前,你需要完全理解一个概念DI(依赖注入)。 当你理解依赖注入以后,对控制反转就非常简......
  • Spring Security 报:Encoded password does not look like BCrypt
    SpringBoot集成Security时,报EncodedpassworddoesnotlooklikeBCrypt原因:SecurityConfig必须Bean的形式实例化/***配置用户身份的configure()方法**@paramauth*@throwsException*/@Overrideprotectedvoidconfigure(AuthenticationManagerBuilderau......
  • SpringBoot读取.yml配置文件最常见的两种方式-源码及其在nacos的应用
    三、第二种方式(推荐)这种方式是小编比较推荐的,虽然看似比​​@Value​​麻烦不少,但是更加的规范,在配合nacos的时候也可以动态的修改,会立即生效,一会小编带大家试一下哈!!为什么推荐这种方式呢,是因为spring他们都是使用这种方式进行配置的,所以跟着官方走不会有错的!!1.修改yml文件我们......
  • SpringCloud学习笔记
    Eureka基本知识Eureka主要学习的是微服务的一些基本概念之类的,至于具体的操作其实都是在配置appolication.yml文件了,多看文档以及自己写过的demo就懂了。Eureka在微服务中承担的角色有三个,一个是注册中心server,一个是服务供给方porvider,以及接受用户请求的consumer,如果从启动类......
  • SpringMVC使用注解开发
    1.编写web.xml(模板代码)ViewCode2.导入springmvc的context和mvc两个依赖,通过context标签可以自动扫描识别包"com.lian.controller"下的所有注解,两个mvc标签是默认配置;context和mvc分别替代了之前的处理器映射器HandleMapper和处理器适配器HandlerAdapter;视图解析器拼接要要跳转......