文章已收录至https://lichong.work,转载请注明原文链接。
ps:欢迎关注公众号“Fun肆编程”或添加我的私人微信交流经验🤝
第一次接触反应式编程可以先看一下反应式宣言,这里有一些名词和概念解释。
简单来说,反应式编程满足以下特质:

经典SpringMVC案例:

如上图所示,在controller中会有一些外部资源调用过程阻塞线程,导致响应延迟,如果阻塞时间长就可能导致用户流失。
tomcat最大线程数配置:2000/s * 0.25s = 500此时,500个线程就可以满足。但如果访问量突增,请求进来就只能排队,这样就会导致响应延迟,用户流失。
Filter/Servlet在生成响应之前可能要等待一些资源的响应以完成请求处理,比如一个jdbc的查询,或者远程服务rpc嗲奥永。在Servlet中阻塞等待是一个低效的操作,这将导致受限系统资源的急剧紧张,比如线程,连接数。
Servlet3.0引入了异步处理请求的能力,使得线程可以不用阻塞等待,提早返回到容器,从而执行更多的请求任务。把耗时的任务提交给另一个异步线程去执行,以及产生响应。
这里自然离不开经典的Spring反应式模型对比

一些Webflux的基础API科普:
webflux中一般用到的api都是Reactor中的概念,符合Reactive Streams规范
当然想充分了解webflux还需要熟悉NIO模型、Reactor模式、lambda与函数式编程等知识,这里就不逐一详解了
传统SpringMVC注解方式实现

基于函数式的的WebFlux开发
其中有两个核心接口:
HandlerFunction:相当于Controller中的具体处理方法,输入为请求,输出封装在Mono中的响应

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

组合使用的话就是如下示例:
先写接口实现:
@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));
}
}
再写对应路由:
@Configuration
public class UserRouterCfg {
public RouterFunction<ServerResponse> userRouterFunctions(UserHandler userHandler) {
return RouterFunctions.route().GET("/user", userHandler::getUser).build();
}
}
官网文档有更多示例可供参考: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();
}
}

// TODO
// 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支持集群环境-设计业务与框架分离及实现定时任务调度
✨欢迎为耿直少年点赞、关注、收藏!!!
👇👇👇