当前位置: 首页 > 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/636973/

相关文章:

  • Mac微信双开
  • 在C语言中
  • 知网AI率60%怎么降?推荐嘎嘎降AI、比话降AI、率零
  • 金融建模新思路:如何用连续时间随机游走(CTRW)预测股价波动?
  • 儿童护眼大路灯哪个牌子好用又便宜?护眼大路灯品牌排行前十名
  • 科技山洪预警监测防汛到户 守护生命防线
  • Linux auditd安全审计实战:从基础配置到高级规则定制
  • CSS如何提高团队协作效率_推广BEM规范减少样式沟通成本
  • 机器学习的一些核心概念
  • CS实验室行业报告:AI算法工程师就业分析报告
  • 影墨·今颜效果展示:弱光环境下的皮肤通透感与阴影层次
  • 【AI知识点】交叉注意力机制:从原理到实战,打通多模态信息交互的桥梁
  • Function Calling与ReAct:Agent工具调用原理
  • MATLAB函数参数验证的妙用
  • Code Interpreter:代码执行型Agent设计
  • STM32开发者必看:5分钟搞定Nanopb协议移植(附常见编译错误解决)
  • 2025-2026年全球吸尘器品牌推荐:五大口碑产品评测对比领先大户型高效除尘场景案例 - 品牌推荐
  • 一场源码泄露事故,验证了怎样的架构设计?
  • 使用LangGraph框架构建大模型Agent:小白程序员必备教程(收藏版)
  • 人机协同智能的瓶颈依然在于休谟与维特根斯坦
  • 每日一题:为什么在 ASP.NET Core 中推荐使用 IAsyncEnumerable,它与 IEnumerable 有什么本质区别?
  • 就在打开游戏的那一刻,弹窗提示“由于找不到vcruntime140.1.dll”——我的解决之旅与经验分享
  • 2025-2026年云南旅行社推荐:五大口碑服务评测对比顶尖定制游信息不透明 - 品牌推荐
  • 人脸识别系统如何合规接入公安视图库?GA/T1400标准避坑指南
  • 3个关键技巧让你轻松掌握ComfyUI-Easy-Use:AI绘画工作流优化神器
  • 深入解析Top K Frequent Elements算法
  • Vue3 + Element Plus 实战:手把手教你封装一个可复用的新手引导组件(附完整代码)
  • SQL在分布式数据库中执行JOIN_数据分片与节点交互原理解析
  • 计量经济学:AI与Python双驱动多源数据处理、机器学习预测及复杂因果识别全流程实战随机森林模型核心技术
  • 2026商业照明设计核心技术与选型全指南:商业照明设计、无主灯照明、景观照明工程、智能照明设计、楼宇照明工程、灯光照明设计选择指南 - 优质品牌商家