从springcloud-gateway了解同步和异步,webflux webMvc、共享变量

webMVC和webFlux

这是spring framework提供的两种不同的Web编程模型

在这里插入图片描述

应用场景:

  • 用 WebMvc:
    项目依赖 Servlet 生态、需要简单同步代码,或使用阻塞式数据库(如 MySQL + JDBC)。

  • 用 WebFlux:
    需要高并发(如每秒万级请求)、低资源消耗,或已使用响应式技术栈(如 MongoDB Reactive、Kafka)。

springcloud -gateway

在 Spring Cloud Gateway 中,请求处理默认是基于 异步非阻塞 模型的,这是由其底层使用的 Reactor Netty 和 WebFlux 框架决定的。

你说是gateway是异步的,但是一个请求不还是要等待下游服务器的响应吗?这不还是同步吗?
nonono~
[EventLoop线程] 接收请求 → 发起网络调用 → 立即释放 → 处理其他请求

[IO完成回调] → 收到响应 → 继续处理
实际上:

  • 同一个线程可以交替处理多个请求的不同阶段
  • 等待IO时不占用线程(线程可以去处理其他请求)
  • 基于回调/事件机制,IO完成后会通知系统

在这里插入图片描述

gateway是基于WebFlux的,理由也很合理:
webMVC 最高只能有200并发,而已经到微服务级别的应用,很明显并发量肯定不会少,网关层面肯定不能用同步去做,异步是最好最确切的选择。

ContextHolder在同步/异步模式中的实现

contextHolder 个人理解,就是一个全局变量,但是这"全局"是‘一个线程’的范围

专业的说: 基于 线程/上下文隔离 的临时数据存储,生命周期与请求或线程绑定

必要性:

  • 解耦业务代码:
    避免在方法参数中层层传递上下文(如用户身份、跟踪ID)。

  • 在拦截器、Service 层、日志工具等地方都能访问统一上下文。

原生Servlet–request.setAttribute

  • Servlet规范本身并未直接定义线程级别的共享变量接口,只定义了会话范围的HttpSession
  • 但是可以通过ServletRequest获取,因为在Servlet规范中,每个请求都是由容器分配的独立线程处理,ServletRequest 和 ServletResponse 对象天然与线程绑定

在这里插入图片描述

tomcat 容器内部 则通过ThreadLocal关联了请求和线程。如果开发者使用ThreadLocal,需要自行清理(存在风险)

WebMvc–ThreadLocal

不同于原生Servlet,这个ThreadLocal不是Tomcat容器给的,而是Springmvc框架本身提供的。

此时: Tomcat 不直接使用 ThreadLocal 存储 Spring MVC 的请求信息,它只是提供了线程模型(每个请求一个线程),而 Spring MVC 在此基础上利用 ThreadLocal 存储请求相关的上下文。

如果自定义使用:

  1. 定义ThreadLocal工具类
public class MyThreadLocalContext {private static final ThreadLocal<String> currentTenantId = new ThreadLocal<>();public static void setTenantId(String tenantId) {currentTenantId.set(tenantId);}public static String getTenantId() {return currentTenantId.get();}public static void clear() {currentTenantId.remove(); // 防止内存泄漏}
}
  1. 在 Filter/Interceptor 中设置和清理

ThreadLocal的隐式线程绑定

在工具类中你会发现,其实set/get过程并没有关联任何的线程ID

这是因为,ThreadLocal内部实现了自动关联线程ID
在这里插入图片描述

WebFlux

ThreadLocal已经无法处理异步请求的问题了,因为异步请求一般都存在线程的切换

springcloud-gateway提供了几种机制来处理这个问题

1. Reactor Context

2. TransmittableThreadLocal

简称 (TTL) 是阿里开源的一个线程本地变量工具,它是对 Java 标准 ThreadLocal 的增强,主要解决了线程池等异步执行场景下的线程变量传递问题。

使用:

  1. 定义TransmittableThreadLocal工具类
package com.ruoyi.common.core.context;import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import com.alibaba.ttl.TransmittableThreadLocal;
import com.ruoyi.common.core.constant.SecurityConstants;
import com.ruoyi.common.core.text.Convert;
import com.ruoyi.common.core.utils.StringUtils;/*** 获取当前线程变量中的 用户id、用户名称、Token等信息 * 注意: 必须在网关通过请求头的方法传入,同时在HeaderInterceptor拦截器设置值。 否则这里无法获取** @author ruoyi*/
public class SecurityContextHolder
{/*** 既然springboot中默认用的是同步请求,必须等待响应完成,为什么这里用异步的threadlocal去存储?** 1.  防御性编程,为未来的异步拓展预留空间** 2.  即时外部请求是重构的,框架内部可能使用线程池处理一些异步任务* 比如日志记录:* @GetMapping("/user/info")*     public UserInfo getUserInfo() {*         // 同步设置用户上下文*         SecurityContextHolder.setUserId("123");**         // 同步操作*         UserInfo info = userService.getInfo();**         // 异步记录日志(很常见的需求)*         logAsync("User info accessed"); // 内部使用线程池,内部会用contextHolder获取到用户信息,**         return info;*     }* 3. springcloud-gateway中请求默认是异步非阻塞的,原因是它是webflux编程*/private static final TransmittableThreadLocal<Map<String, Object>> THREAD_LOCAL = new TransmittableThreadLocal<>();public static void set(String key, Object value){Map<String, Object> map = getLocalMap();map.put(key, value == null ? StringUtils.EMPTY : value);}public static String get(String key){Map<String, Object> map = getLocalMap();return Convert.toStr(map.getOrDefault(key, StringUtils.EMPTY));}public static <T> T get(String key, Class<T> clazz){Map<String, Object> map = getLocalMap();return StringUtils.cast(map.getOrDefault(key, null));}public static Map<String, Object> getLocalMap(){Map<String, Object> map = THREAD_LOCAL.get();if (map == null){map = new ConcurrentHashMap<String, Object>();THREAD_LOCAL.set(map);}return map;}public static void setLocalMap(Map<String, Object> threadLocalMap){THREAD_LOCAL.set(threadLocalMap);}public static Long getUserId(){return Convert.toLong(get(SecurityConstants.DETAILS_USER_ID), 0L);}public static void setUserId(String account){set(SecurityConstants.DETAILS_USER_ID, account);}public static String getUserName(){return get(SecurityConstants.DETAILS_USERNAME);}public static void setUserName(String username){set(SecurityConstants.DETAILS_USERNAME, username);}public static String getUserKey(){return get(SecurityConstants.USER_KEY);}public static void setUserKey(String userKey){set(SecurityConstants.USER_KEY, userKey);}public static String getPermission(){return get(SecurityConstants.ROLE_PERMISSION);}public static void setPermission(String permissions){set(SecurityConstants.ROLE_PERMISSION, permissions);}public static void remove(){THREAD_LOCAL.remove();}
}
  1. 在拦截器中添加和删除
/*** 自定义请求头拦截器,将Header数据封装到线程变量中方便获取* 注意:此拦截器会同时验证当前用户有效期自动刷新有效期** @author ruoyi*/
public class HeaderInterceptor implements AsyncHandlerInterceptor
{@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception{if (!(handler instanceof HandlerMethod)){return true;}SecurityContextHolder.setUserId(ServletUtils.getHeader(request, SecurityConstants.DETAILS_USER_ID));SecurityContextHolder.setUserName(ServletUtils.getHeader(request, SecurityConstants.DETAILS_USERNAME));SecurityContextHolder.setUserKey(ServletUtils.getHeader(request, SecurityConstants.USER_KEY));String token = SecurityUtils.getToken();if (StringUtils.isNotEmpty(token)){LoginUser loginUser = AuthUtil.getLoginUser(token);if (StringUtils.isNotNull(loginUser)){AuthUtil.verifyLoginUserExpire(loginUser);SecurityContextHolder.set(SecurityConstants.LOGIN_USER, loginUser);}}return true;}@Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)throws Exception{SecurityContextHolder.remove();}
}

TTL底层依赖ThreadLocal,但是它的核心是 跨线程传递 ThreadLocal 值
具体怎么做到的:父子线程之间的值传递

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.tpcf.cn/bicheng/89187.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何在 Pytest 中调用其他用例返回的接口参数?

回答重点在 Pytest 中&#xff0c;我们可以通过使用共享夹具&#xff08;fixtures&#xff09;来调用和复用其他用例返回的接口参数。在 Pytest 中&#xff0c;fixtures 提供了一种灵活且有组织的方式来共享测试数据或对象。具体步骤如下&#xff1a;1&#xff09;首先&#xf…

倒计时熔断机制的出价逻辑

一、业务背景传统竞价机制中&#xff0c;“倒计时结束”是系统决定成交者的关键逻辑&#xff0c;但在实际中&#xff0c;最后3秒突然被抢价的情况极为常见&#xff0c;出现以下问题&#xff1a;用户投诉平台机制不公平&#xff1b;用户出价但未成交&#xff0c;产生争议订单&am…

未来手机会自动充电吗

未来手机实现‌全自动充电&#xff08;无需人为干预&#xff09;‌是技术发展的明确趋势&#xff0c;目前已有部分技术落地&#xff0c;但要达到“随时随地无感补电”&#xff0c;仍需突破以下关键领域&#xff1a;一、已实现的技术&#xff08;当下可用的“半自动”充电&#…

MySQL高级篇(二):深入理解数据库事务与MySQL锁机制

引言在现代数据库系统中&#xff0c;事务和锁机制是确保数据一致性和完整性的两大核心技术。无论是金融交易系统、电商平台还是企业级应用&#xff0c;都离不开这些基础功能的支持。本文将全面剖析数据库事务的四大特性&#xff0c;深入探讨MySQL中的各种锁机制&#xff0c;帮助…

XML 指南

XML 指南 引言 XML(可扩展标记语言)是一种用于存储和传输数据的标记语言,它具有高度的可扩展性和灵活性。在互联网和软件开发领域,XML被广泛应用于数据交换、配置文件、文档存储等场景。本文将为您详细介绍XML的基本概念、语法规则、应用场景以及开发技巧,帮助您全面了解…

Flink Watermark原理与实战

一、引言Flink 作为一款强大的流处理框架&#xff0c;在其中扮演着关键角色。今天&#xff0c;咱们来聊聊 Flink 中一个极为重要的概念 —— Watermark&#xff08;水位线&#xff09;&#xff0c;它是处理乱序数据和准确计算的关键。接下来我们直入主题&#xff0c;首先来看看…

Rust Web 全栈开发(五):使用 sqlx 连接 MySQL 数据库

Rust Web 全栈开发&#xff08;五&#xff09;&#xff1a;使用 sqlx 连接 MySQL 数据库Rust Web 全栈开发&#xff08;五&#xff09;&#xff1a;使用 sqlx 连接 MySQL 数据库项目创建数据库准备连接请求功能实现Rust Web 全栈开发&#xff08;五&#xff09;&#xff1a;使用…

【zynq7020】PS的“Hello World”

目录 基本过程 新建Vivado工程 ZYNQ IP核设置 使用SDK进行软件开发 基于Vivado2017 Vivado工程建立 SDK调试 固化程序 注&#xff1a;Vivado 2019.1 及之前&#xff1a;默认使用 SDK Vivado 2019.2-2020.1&#xff1a;逐步过渡&#xff0c;支持 SDK 与 Vitis 并存 Vi…

希尔排序和选择排序及计数排序的简单介绍

希尔排序法又称缩小增量法。希尔排序法的基本思想是&#xff1a;先选定一个整数gap&#xff0c;把待排序文件中所有数据分成几个组&#xff0c;所有距离为gap的数据分在同一组内&#xff0c;并对每一组内的数据进行排序。然后gap减减&#xff0c;重复上述分组和排序的工作。当到…

Solid Edge多项目并行,浮动许可如何高效调度?

在制造企业的数字化设计体系中&#xff0c;Solid Edge 作为主流 CAD 工具&#xff0c;因其灵活的建模能力、同步技术和强大的装配设计功能&#xff0c;广泛应用于机械设备、零部件制造等行业的研发场景。随着企业设计任务复杂化&#xff0c;多项目并行成为常态&#xff0c;Soli…

Flink cdc 使用总结

Flink 与 Flink CDC 版本兼容对照表Flink 版本支持的 Flink CDC 版本关键说明Flink 1.11.xFlink CDC 1.2.x早期版本&#xff0c;需注意 Flink 1.11.0 的 Bug&#xff08;如 Upsert 写入问题&#xff09;&#xff0c;建议使用 1.11.1 及以上。Flink 1.12.xFlink CDC 2.0.x&#…

企业培训笔记:axios 发送 ajax 请求

文章目录axios 简介一&#xff0c;Vue工程中安装axios二&#xff0c;编写app.vue三&#xff0c;编写HomeView.vue四&#xff0c;Idea打开后台项目五&#xff0c;创建HelloController六&#xff0c;配置web访问端口七&#xff0c;运行项目&#xff0c;查看效果&#xff08;一&am…

Maven下载与配置对Java项目的理解

目录 一、背景 二、JAVA项目与Maven的关系 2.1标准java项目 2.2 maven 2.2.1 下载maven 1、下载 2、配置环境 2.2.2 setting.xml 1、配置settings.xml 2、IDEA配置maven 一、背景 在java项目中&#xff0c;新手小白很有可能看不懂整体的目录结构&#xff0c;以及每个…

Mars3d的走廊只能在一个平面的无法折叠的解决方案

问题场景&#xff1a;1. Mars3d的CorridorEntity只能在一个平面修改高度值&#xff0c;无法根据坐标点位制作有高度值的走廊效果&#xff0c;想要做大蜀山盘山走廊的效果实现不了。解决方案&#xff1a;1.使用原生cesium实现对应的走廊的截面形状、走廊的坐标点&#xff0c;包括…

LeetCode 每日一题 2025/7/7-2025/7/13

记录了初步解题思路 以及本地实现代码&#xff1b;并不一定为最优 也希望大家能一起探讨 一起进步 目录7/7 1353. 最多可以参加的会议数目7/8 1751. 最多可以参加的会议数目 II7/9 3439. 重新安排会议得到最多空余时间 I7/10 3440. 重新安排会议得到最多空余时间 II7/11 3169. …

Bash常见条件语句和循环语句

以下是 Bash 中常用的条件语句和循环语句分类及语法说明&#xff0c;附带典型用例&#xff1a;一、条件语句 1. if 语句 作用&#xff1a;根据条件执行不同代码块 语法&#xff1a; if [ 条件 ]; then# 条件为真时执行 elif [ 其他条件 ]; then# 其他条件为真时执行 else# 所有…

uni-app 选择国家区号

uni-app选择国家区号组件 hy-countryPicker 我们在做登录注册功能的时候&#xff0c;可能会遇到选择区号来使用不同国家手机号来登录或者注册的功能。这里我就介绍下我这个uni-app中使用的选择区号的组件&#xff0c;包含不同国家国旗图标。 效果图 别的不说&#xff0c;先来…

客户端主机宕机,服务端如何处理 TCP 连接?详解

文章目录一、客户端主机宕机后迅速重启1、服务端有数据发送2、服务端开启「保活」机制3、服务端既没有数据发送&#xff0c;也没有开启「保活」机制二、客户端主机宕机后一直没有重启1、服务端有数据发送2、服务端开启「保活」机制3、服务端既没有数据发送&#xff0c;也没有开…

《大数据技术原理与应用》实验报告五 熟悉 Hive 的基本操作

目 录 一、实验目的 二、实验环境 三、数据集 四、实验内容与完成情况 4.1 创建一个内部表 stocks&#xff0c;字段分隔符为英文逗号&#xff0c;表结构下所示。 4.2 创建一个外部分区表 dividends&#xff08;分区字段为 exchange 和symbol&#xff09;&#xff0c;字段…

【橘子分布式】Thrift RPC(编程篇)

一、简介 之前我们研究了一下thrift的一些知识&#xff0c;我们知道他是一个rpc框架&#xff0c;他作为rpc自然是提供了客户端到服务端的访问以及两端数据传输的消息序列化&#xff0c;消息的协议解析和传输&#xff0c;所以我们今天就来了解一下他是如何实现这些功能&#xff…