码农知识堂 - 1000bd
  •   Python
  •   PHP
  •   JS/TS
  •   JAVA
  •   C/C++
  •   C#
  •   GO
  •   Kotlin
  •   Swift
  • 【Java-webflux】Spring5新特性之webflux反应式编程-Project Reactor


    文章已收录至https://lichong.work,转载请注明原文链接。
    ps:欢迎关注公众号“Fun肆编程”或添加我的私人微信交流经验🤝

    概念

    第一次接触反应式编程可以先看一下反应式宣言,这里有一些名词和概念解释。
    简单来说,反应式编程满足以下特质:

    1. 即时响应性:只要建立连接, 系统就会及时地做出响应。
    2. 回弹性:系统在出现失败时依然保持即时响应性。
    3. 弹性:系统在不断变化的工作负载之下依然保持即时响应性,也就是说可以弹性伸缩。
    4. 消息驱动:反应式系统依赖异步的消息传递,从而确保了松耦合、隔离、位置透明的组件之间有着明确边界。

    反应式宣言

    一、现在-流行的SpringMVC所遇到的困境

    经典SpringMVC案例:
    经典SpringMVC案例
    如上图所示,在controller中会有一些外部资源调用过程阻塞线程,导致响应延迟,如果阻塞时间长就可能导致用户流失。

    二、过渡-Servlet3.0异步请求原理

    1.Servlet同步请求问题

    • 假设同步场景:
      平均访问量:2000/s
      请求耗时:250ms
      tomcat最大线程数配置:2000/s * 0.25s = 500

    此时,500个线程就可以满足。但如果访问量突增,请求进来就只能排队,这样就会导致响应延迟,用户流失。

    2.Servlet异步如何解决

    Filter/Servlet在生成响应之前可能要等待一些资源的响应以完成请求处理,比如一个jdbc的查询,或者远程服务rpc嗲奥永。在Servlet中阻塞等待是一个低效的操作,这将导致受限系统资源的急剧紧张,比如线程,连接数。
    Servlet3.0引入了异步处理请求的能力,使得线程可以不用阻塞等待,提早返回到容器,从而执行更多的请求任务。把耗时的任务提交给另一个异步线程去执行,以及产生响应。

    • 异步场景:
      业务平均访问量:5000/s
      同步请求的耗时:250ms
      异步请求的耗时:20ms
      理论同步场景下tomcat最大线程数配置:5000/s * 0.25s = 1250
      但异步时这里tomcat最大线程数我们可以使用100,此时我们使用Servlet异步线程,将业务逻辑放到一个其他线程池中的线程去执行,不阻塞tomcat处理请求。同时可以根据不同业务使用不同线程池,从而达到对指定业务限流的效果。

    三、未来-WebFlux反应式模型

    这里自然离不开经典的Spring反应式模型对比
    Spring响应式模型对比

    • Spring MVC
      • 构建于 Servlet API 之上
      • 同步阻塞 I/O 模型, 认为应用汇阻塞当前线程,所以一个 Request 对应一个 Thread,需要有一个含有大量线程的线程池
    • Spring WebFlux
      • 构建于 Reactive Streams Adapters 之上
      • 异步非阻塞 I/O 模型,认为应用不会阻塞当前线程,所以只是需要一个包含少数固定线程数的线程池 (event loop workers) 来处理请求
      • Spring反应式模型中,完全支持了全链路的流式传输,从而实现【浏览器-中间件-程序-缓存/数据库】的反应式编程。

    四、从0到1快速实战Webflux

    一些Webflux的基础API科普:
    webflux中一般用到的api都是Reactor中的概念,符合Reactive Streams规范

    • Reactor 的主要模块
      Reactor 框架主要有两个主要的模块:reactor-core 和 reactor-ipc。前者主要负责 Reactive Programming 相关的核心 API 的实现,后者负责高性能网络通信的实现,目前是基于 Netty 实现的。
    • Reactor 的主要类
      在 Reactor 中,经常使用的类并不是很多,主要有以下两个:
      • Mono 实现了 org.reactivestreams.Publisher 接口,代表0到1个元素的发布者。
      • Flux 同样实现了 org.reactivestreams.Publisher 接口,代表0到N个元素的发布者。
    • 可能会使用到的类
      • Scheduler 表示背后驱动反应式流的调度器,通常由各种线程池实现。

    当然想充分了解webflux还需要熟悉NIO模型、Reactor模式、lambda与函数式编程等知识,这里就不逐一详解了

    1.接口实现webflux的两种方式

    • 传统SpringMVC注解方式实现
      传统SpringMVC注解方式实现

    • 基于函数式的的WebFlux开发
      其中有两个核心接口:

      • HandlerFunction:相当于Controller中的具体处理方法,输入为请求,输出封装在Mono中的响应
        HandlerFunction

      • RouterFunction:相当于RequestMapping,将url映射到具体的HandlerFunction,输入为请求,输出封装在Mono中的HandlerFunction
        RouterFunction

    组合使用的话就是如下示例:
    先写接口实现:

    @Component
    public class UserHandler {
        
        public Mono<ServerResponse> getUser(ServerRequest request) {
            Optional<String> userId = request.queryParam("userId");
            return userId.map(id ->
                            ServerResponse.ok().contentType(MediaType.APPLICATION_JSON).body(Mono.just(AuthInfo.builder().id(id).appName("lichong.work").build()), AuthInfo.class))
                    .orElseGet(() ->
                            ServerResponse.badRequest().contentType(MediaType.TEXT_PLAIN).body(Mono.just("userId is empty"), String.class));
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    再写对应路由:

    @Configuration
    public class UserRouterCfg {
    
        public RouterFunction<ServerResponse> userRouterFunctions(UserHandler userHandler) {
            return RouterFunctions.route().GET("/user", userHandler::getUser).build();
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    官网文档有更多示例可供参考:API文档,在RouterFunction中还可以增加一些过滤器,都是函数式的流式编程实现,自己尝试一下就可以很快理解了,这里我简单写了一个示例参考:

    @Configuration
    public class UserRouterCfg {
    
        public RouterFunction<ServerResponse> userRouterFunctions(UserHandler userHandler) {
            return RouterFunctions.route().filter((((serverRequest, handlerFunction) -> {
                ServerRequest.Headers headers = serverRequest.headers();
                List<String> tokenHeader = headers.header("token");
                if (tokenHeader.isEmpty()) {
                    return ServerResponse.status(HttpStatus.FORBIDDEN).build();
                }
                return handlerFunction.handle(serverRequest);
            }))).GET("/user", userHandler::getUser).build();
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    2.自定义全局异常

    自定义全局异常

    3. webflux和关系型数据库整合

    // TODO

    五、Project Reactor项目

    // TODO

    文章已收录至https://lichong.work,转载请注明原文链接。
    ps:欢迎关注公众号“Fun肆编程”或添加我的私人微信交流经验🤝

    ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~往期精选🪶~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

    【Docker】入门教程-基本概念解读
    【前端-开发环境】使用NVM实现不同nodejs版本的自由切换(NVM完整安装使用手册)
    【前端-NPM私服】内网使用verdaccio搭建私有npm服务器
    【前端-IE兼容】Win10和Win11使用Edge调试前端兼容IE6、IE7、IE8、IE9、IE10、IE11问题
    【前端-工程化】React项目工程化记录-内置项目活文档(老项目升级优化-集成Hosky/ESLint/Prettier-升级Webpack/Babel/NodeSass/React)
    【工具-TWRP-frp-Termux】旧手机暴改成免费云服务器-MIUI刷TWRP安装magisk获取root
    【工具-Shell脚本】java程序产品包模板-linux和windows通用shell启动停止脚本(无需系统安装Java运行环境)
    【工具-Nginx】从入门安装到高可用集群搭建
    【工具-Nginx】Nginx高性能通用配置文件-注释版-支持防刷限流、可控高并发、HTTP2、防XSS、Gzip、OCSP Stapling、负载、SSL
    【工具-WireShark】网络HTTP抓包使用教程
    【后端-maven打包】通过profile标签解决同时打jar包 war包需求
    【架构-DDD】使用领域驱动设计-互联网未来架构设计之道(一)
    【后端-SpringCache】基于Spring Cache封装一个能够批量操作的Redis缓存记录下踩坑历程(pipeline或mget封装)
    【后端-SkyWalking】SkyWalking前后端开发环境搭建详细教程步骤-6.x/7.x/8.x版本通用-插件二次开发利器(一)
    【后端-Quartz】Springboot整合Quartz支持集群环境-设计业务与框架分离及实现定时任务调度

    ✨欢迎为耿直少年点赞、关注、收藏!!!

    👇👇👇

  • 相关阅读:
    免费开源的非标项目型制造BOM一键导入方案介绍
    MySQL高级SQL语句(二)
    Redis缓存——快速入门
    时间复杂度
    ERP编制物料清单 基础
    dockerfile 搭建lnmp+wordpress,docker-compose搭建lnmp+wordpress
    ch0_OSI 七层网络协议介绍
    连续1D空战辅助决策问题的DDPG实现
    Win11环境Mecab日语分词和词性分析以及动态库DLL not found问题(Python3.10)
    资源、死锁、如何监测死锁
  • 原文地址:https://blog.csdn.net/lc1025082182/article/details/126447895
  • 最新文章
  • 攻防演习之三天拿下官网站群
    数据安全治理学习——前期安全规划和安全管理体系建设
    企业安全 | 企业内一次钓鱼演练准备过程
    内网渗透测试 | Kerberos协议及其部分攻击手法
    0day的产生 | 不懂代码的"代码审计"
    安装scrcpy-client模块av模块异常,环境问题解决方案
    leetcode hot100【LeetCode 279. 完全平方数】java实现
    OpenWrt下安装Mosquitto
    AnatoMask论文汇总
    【AI日记】24.11.01 LangChain、openai api和github copilot
  • 热门文章
  • 十款代码表白小特效 一个比一个浪漫 赶紧收藏起来吧!!!
    奉劝各位学弟学妹们,该打造你的技术影响力了!
    五年了,我在 CSDN 的两个一百万。
    Java俄罗斯方块,老程序员花了一个周末,连接中学年代!
    面试官都震惊,你这网络基础可以啊!
    你真的会用百度吗?我不信 — 那些不为人知的搜索引擎语法
    心情不好的时候,用 Python 画棵樱花树送给自己吧
    通宵一晚做出来的一款类似CS的第一人称射击游戏Demo!原来做游戏也不是很难,连憨憨学妹都学会了!
    13 万字 C 语言从入门到精通保姆级教程2021 年版
    10行代码集2000张美女图,Python爬虫120例,再上征途
Copyright © 2022 侵权请联系2656653265@qq.com    京ICP备2022015340号-1
正则表达式工具 cron表达式工具 密码生成工具

京公网安备 11010502049817号