当前位置: 首页 > news >正文

DotNetPy:现代.NET 与 Python 互操作 实战指南捉

我为什么会发出这个疑问呢?是因为我研究Web开发中的一个问题时,HTTP请求体在 Filter(过滤器)处被读取了之后,在 Controller(控制层)就读不到值了,使用 @RequestBody 的时候。

无论是字节流(InputStream / OutputStream)还是字符流(Reader / Writer),所有基于流的读取操作都会维护一个 "位置指针"。

初始状态下,指针指向流的起始位置(position = 0);

每次调用 read() / read(byte[]) / read(char[]) 等读取方法时,指针会向后移动对应字节数;

当指针移动到流的末尾(没有更多数据),read() 方法会返回 -1,表示流读取完毕;

指针移动后不会自动回退,也无法反向移动(除非流显式支持重置),因此再次读取只能得到 -1。

类比:IO 流的读取过程,就像用 磁带播放器听磁带 —— 磁头(对应流的位置指针)从磁带开头(指针 0)开始移动,每读一个字节 / 字符,磁头就往后走一步;当磁头走到磁带末尾,再继续播放(读取)就只能听到 "沙沙声"(流返回 -1),并且磁头不会自动回到开头。

当然,不是所有流都只能读一次,基于内存的流(如 ByteArrayInputStream / CharArrayReader)支持重置指针,因为它们的数据源是内存中的数组(数据不会消失),可以通过 mark() 和 reset() 方法将指针 恢复 到标记位置。

需要注意:

调用 reset() 前必须先调用 mark(int readlimit);

不是所有流都支持 mark() / reset(),可以通过 inputStream.markSupported() 来进行判断。

使用 mark() 和 reset() 方法:

// 仅适用于支持mark的流

public void processWithMark(InputStream input) throws IOException {

if (!input.markSupported()) {

throw new IOException("Mark not supported");

}

// 标记当前位置,参数100表示最多可回退100字节

input.mark(100);

// 第一次读取

byte[] firstRead = new byte[50];

input.read(firstRead);

System.out.println("First read: " + new String(firstRead));

// 重置到标记位置

input.reset();

// 第二次读取(相同内容)

byte[] secondRead = new byte[50];

input.read(secondRead);

System.out.println("Second read: " + new String(secondRead));

}

使用 包装类 解决上文我们提到的 HTTP请求体多次读取 的问题:

public class MyRequestWrapper extends HttpServletRequestWrapper {

private final byte[] body; // 缓存请求体的字节数组

public MyRequestWrapper(HttpServletRequest request) throws IOException {

super(request);

// 关键步骤:在构造时一次性读取并存储原始请求流

body = StreamUtils.copyToByteArray(request.getInputStream());

}

// 提供一个便捷方法,用于在过滤器中获取请求体内容(例如记录日志)

// 使用时,直接调用 getBodyString() 即可

public String getBodyString() throws UnsupportedEncodingException {

return new String(body, this.getCharacterEncoding());

}

@Override

public ServletInputStream getInputStream() throws IOException {

// 每次调用都返回一个基于缓存数据的新流

ByteArrayInputStream bais = new ByteArrayInputStream(body);

return new ServletInputStream() {

@Override

public int read() {

return bais.read();

}

@Override

public boolean isFinished() {

return bais.available() == 0;

}

@Override

public boolean isReady() {

return true;

}

@Override

public void setReadListener(ReadListener readListener) {

// 无需实现

}

};

}

@Override

public BufferedReader getReader() throws IOException {

return new BufferedReader(new InputStreamReader(this.getInputStream(), this.getCharacterEncoding()));

}

}

然后在 过滤器 处包装请求:

@Slf4j

@Configuration

public class RequestCachingFilterConfig {

@Bean

public FilterRegistrationBean requestCachingFilter() {

FilterRegistrationBean registrationBean = new FilterRegistrationBean();

// 核心:创建过滤器,包装请求为 ContentCachingRequestWrapper

registrationBean.setFilter(new OncePerRequestFilter() {

@Override

protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)

throws ServletException, IOException {

// 1. 仅包装 HTTP 请求(排除 WebSocket 等)

if (request instanceof HttpServletRequest && !(request instanceof ContentCachingRequestWrapper)) {

log.info("==========进入requestCachingFilter========");

// 2. 包装请求(自动缓存请求体)

MyRequestWrapper wrappedRequest = new MyRequestWrapper(request);

filterChain.doFilter(wrappedRequest, response); // 传递包装后的请求

} else {

filterChain.doFilter(request, response); // 无需包装,直接放行

}

}

});

// 3. 配置拦截所有请求(可根据需求调整 URL 模式)

registrationBean.addUrlPatterns("/*");

registrationBean.setOrder(1); // 优先级最高,确保先于其他过滤器执行

registrationBean.setName("requestCachingFilter");

return registrationBean;

}

}

IO 流只能读取一次,是 精心设计的,贴合操作系统文件 / 网络 IO 的 "顺序消费" 特性,保持和底层系统的一致性。俦沤瘸雀

http://www.jsqmd.com/news/609022/

相关文章:

  • KIHU快狐|49寸户外触摸查询机3000亮度银行用
  • 【PyO3 × GraalVM × CPython 3.14原生AOT三重验证】:2026唯一通过PEP 718认证的配置流程
  • Lobe Theme 国际化支持:如何为你的语言贡献翻译
  • AI + Cybersecurity
  • 虚拟线程调度失灵、协程泄漏、监控断连——Java 25高并发架构崩塌前的5个预警信号,速查!
  • 别再死记硬背公式了!用MATLAB Simulink从零搭建一阶倒立摆模型(附完整.m文件)
  • 新手避坑指南:用Seurat分析单细胞数据时,这5个参数设置错误最要命
  • 三步掌握FullCalendar Vue3组件:从入门到场景化落地
  • 如何让求职效率提升300%?NewJob智能插件帮你避开90%的无效岗位
  • ESP32-CAMERA官方例程在S3开发板上不工作?手把手教你排查引脚与PSRAM配置
  • 谷歌 2026-完整的 AI 帝国蓝图
  • 开源项目管理工具Taskcafe完整贡献指南:7步加入看板协作开发
  • gh_mirrors/resum/resume字体系统详解:Adobe中文字体与FontAwesome图标集成
  • 线性代数别死记!用Python的NumPy库5分钟搞定向量线性相关性判断
  • Blue Topaz主题:10分钟打造你的专属Obsidian蓝色笔记空间
  • doT.js测试终极指南:如何编写高质量的模板测试用例
  • AD9361驱动移植避坑指南:如何用Vivado TCL脚本为你的自定义板卡快速适配官方HDL代码
  • 别再手动拖拽了!用Next AI Draw.io + Claude Sonnet 4.5,一句话生成AWS架构图
  • VNC Viewer连接CentOS 8的完整指南:解决黑屏与画质问题
  • 终极指南:FPSSample大型Unity项目管理实践与协作方法
  • C#(CShape)基础语法
  • Sonic云真机平台测试用例管理:公共步骤与模块化设计思想
  • 别再只玩ChatGPT了!手把手带你用PyTorch和Isaac Sim复现一个能‘看、说、动’的VLA机器人Demo
  • Stable Diffusion 入门:架构、空间与生成流程概览
  • 避坑指南:YOLOv11转ONNX模型时,为什么必须先卸载ultralytics库?
  • iFakeLocation:跨平台iOS虚拟定位开源工具的全方位实践指南
  • 痞子衡嵌入式:turbo-spiboot - 一种基于MCUBoot协议的二级SPI加载APP提速方案匣
  • Android组件参数传递终极指南:Fragment与Activity通信的10个最佳实践
  • 分钟搞懂深度学习AI:实操篇:Attention镭
  • 终极Windows驱动清理指南:DriverStore Explorer轻松释放20GB磁盘空间