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

eternal_silence

我为什么会发出这个疑问呢是因为我研究Web开发中的一个问题时请求体在 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));}使用 包装类 解决上文我们提到的 请求体多次读取 的问题public class MyRequestWrapper extends ServletRequestWrapper {private final byte[] body; // 缓存请求体的字节数组public MyRequestWrapper(ServletRequest request) throws IOException {super(request);// 关键步骤在构造时一次性读取并存储原始请求流body StreamUtils.copyToByteArray(request.getInputStream());}// 提供一个便捷方法用于在过滤器中获取请求体内容例如记录日志// 使用时直接调用 getBodyString() 即可public String getBodyString() throws UnsupportedEncodingException {return new String(body, this.getCharacterEncoding());}Overridepublic ServletInputStream getInputStream() throws IOException {// 每次调用都返回一个基于缓存数据的新流ByteArrayInputStream bais new ByteArrayInputStream(body);return new ServletInputStream() {Overridepublic int read() {return bais.read();}Overridepublic boolean isFinished() {return bais.available() 0;}Overridepublic boolean isReady() {return true;}Overridepublic void setReadListener(ReadListener readListener) {// 无需实现}};}Overridepublic BufferedReader getReader() throws IOException {return new BufferedReader(new InputStreamReader(this.getInputStream(), this.getCharacterEncoding()));}}然后在 过滤器 处包装请求Slf4jConfigurationpublic class RequestCachingFilterConfig {Beanpublic FilterRegistrationBean requestCachingFilter() {FilterRegistrationBean registrationBean new FilterRegistrationBean();// 核心创建过滤器包装请求为 ContentCachingRequestWrapperregistrationBean.setFilter(new OncePerRequestFilter() {Overrideprotected void doFilterInternal(ServletRequest request, ServletResponse response, FilterChain filterChain)throws ServletException, IOException {// 1. 仅包装 请求排除 WebSocket 等if (request instanceof ServletRequest !(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 的 顺序消费 特性保持和底层系统的一致性。媚镭拐盘

相关文章:

eternal_silence

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

JoyCon-Driver完整指南:如何在Windows上免费使用Switch手柄玩PC游戏

JoyCon-Driver完整指南:如何在Windows上免费使用Switch手柄玩PC游戏 【免费下载链接】JoyCon-Driver A vJoy feeder for the Nintendo Switch JoyCons and Pro Controller 项目地址: https://gitcode.com/gh_mirrors/jo/JoyCon-Driver 想让你的任天堂Switch …...

Transformers.js:在浏览器中运行200+AI模型的革命性突破

Transformers.js:在浏览器中运行200AI模型的革命性突破 【免费下载链接】transformers.js State-of-the-art Machine Learning for the web. Run 🤗 Transformers directly in your browser, with no need for a server! 项目地址: https://gitcode.co…...

3步轻松备份QQ空间所有历史说说:GetQzonehistory完整指南

3步轻松备份QQ空间所有历史说说:GetQzonehistory完整指南 【免费下载链接】GetQzonehistory 获取QQ空间发布的历史说说 项目地址: https://gitcode.com/GitHub_Trending/ge/GetQzonehistory 你是否担心QQ空间里那些记录青春岁月的说说、照片和评论会随着时间…...

[具身智能-464]:语音识别与语音合成的关键和核心是模型文件,分别阐述它们的输入和输出

在深度学习时代,模型文件(通常包含网络结构定义和训练好的权重参数)确实是整个系统的灵魂。它就像是经过数年苦读后,存储在人脑中的“知识”和“经验”。没有模型文件,代码只是一堆空壳;有了模型文件&#…...

国产芯片适配进度告急!MCP 2026强制认证倒计时180天,你还在用X86测试环境凑合?

更多请点击: https://intelliparadigm.com 第一章:MCP 2026强制认证政策全景解读 MCP(Model Certification Protocol)2026 是由全球人工智能治理联盟(GAIG)于2024年10月正式发布的下一代大模型合规性认证框…...

基于GCP的云原生AI智能体快速部署:基础设施即代码实践指南

1. 项目概述:一个云原生智能体的“样板间”最近在折腾云原生和AI应用开发,发现很多朋友想把手头的AI模型或者智能体(Agent)部署到云端,但往往卡在第一步:环境搭建和基础架构配置。这让我想起了自己刚开始接…...

[具身智能-462]:语音识别是把通过麦克风接收到的声波转化成语音波形,经过数字化后的语音文件转化成文字;语音合成是把文字转换成语音波形,然后通过speaker转换成声波。

人机语音交互中“听”与“说”的完整闭环:语音识别 (ASR):是“听”的过程,即 声波 →→ 数字信号 →→ 文字。语音合成 (TTS):是“说”的过程,即 文字 →→ 数字信号 →→ 声波。为了更透彻地理解这两个过程背后的技术…...

Docker运行AI代码为何总崩溃?揭秘沙箱隔离4大配置陷阱及3分钟修复方案

更多请点击: https://intelliparadigm.com 第一章:Docker运行AI代码崩溃现象与沙箱隔离本质剖析 当在 Docker 容器中运行 PyTorch 或 TensorFlow 训练脚本时,常出现进程静默退出、CUDA 初始化失败或 SIGSEGV 段错误——这些并非单纯代码缺陷…...

机器学习核心概念与实战技巧解析

1. 机器学习核心概念解析作为一名从业多年的数据科学家,我经常被问到"如何快速理解机器学习的关键概念"。今天我就用最直白的方式,带大家拆解10个最核心的术语。这些不是教科书定义,而是我每天实际工作中都在使用的实战理解。机器学…...

如何让经典游戏在现代显示器上完美呈现?PvZWidescreen模组的技术解析

如何让经典游戏在现代显示器上完美呈现?PvZWidescreen模组的技术解析 【免费下载链接】PvZWidescreen Widescreen mod for Plants vs Zombies 项目地址: https://gitcode.com/gh_mirrors/pv/PvZWidescreen 对于许多经典游戏爱好者来说,《植物大战…...

如何构建专业级设计系统:Outfit字体9字重开源解决方案技术架构指南

如何构建专业级设计系统:Outfit字体9字重开源解决方案技术架构指南 【免费下载链接】Outfit-Fonts The most on-brand typeface 项目地址: https://gitcode.com/gh_mirrors/ou/Outfit-Fonts Outfit字体是一款专为品牌自动化设计的开源几何无衬线字体&#xf…...

前端GIF处理效率提升300%?gifuct-js深度解析与应用实践

前端GIF处理效率提升300%?gifuct-js深度解析与应用实践 【免费下载链接】gifuct-js Fastest javascript .GIF decoder/parser 项目地址: https://gitcode.com/gh_mirrors/gi/gifuct-js 在现代前端开发中,GIF动态图像的处理一直是个技术挑战。传统…...

2026年人工智能论文降AI工具推荐:算法研究和模型分析部分降AI方案

2026年人工智能论文降AI工具推荐:算法研究和模型分析部分降AI方案 研究生群里聊起AI率的问题,发现十个人里起码六七个都在用工具降。主流的选择其实就那几款,关键是选对了能省很多麻烦。 综合价格和效果,我主推嘎嘎降AI&#xf…...

LibreOffice Online如何实现企业级文档协作?深度解析架构设计与性能调优

LibreOffice Online如何实现企业级文档协作?深度解析架构设计与性能调优 【免费下载链接】online Read-only Mirror - no pull request (use https://gerrit.libreoffice.org instead) 项目地址: https://gitcode.com/gh_mirrors/onl/online 面对企业数字化转…...

如何用WebToEpub将网页小说永久保存为电子书:完整指南

如何用WebToEpub将网页小说永久保存为电子书:完整指南 【免费下载链接】WebToEpub A simple Chrome (and Firefox) Extension that converts Web Novels (and other web pages) into an EPUB. 项目地址: https://gitcode.com/gh_mirrors/we/WebToEpub 还在为…...

零成本打造专业4K播放器:创维E900V22C电视盒子终极改造指南

零成本打造专业4K播放器:创维E900V22C电视盒子终极改造指南 【免费下载链接】e900v22c-CoreELEC Build CoreELEC for Skyworth e900v22c 项目地址: https://gitcode.com/gh_mirrors/e9/e900v22c-CoreELEC 想将闲置的创维E900V22C电视盒子变身为强大的4K媒体播…...

嵌入式系统ACPI电源管理技术解析与实践

1. 嵌入式系统电源管理概述在嵌入式系统设计中,电源管理始终是一个关键挑战。随着Intel架构在嵌入式领域的广泛应用,从工业控制设备到便携式医疗仪器,再到智能交通系统,对能效的要求越来越高。我曾参与过一个基于Intel Atom处理器…...

Reference Extractor:当学术文献意外丢失时,如何3分钟内找回所有引用?

Reference Extractor:当学术文献意外丢失时,如何3分钟内找回所有引用? 【免费下载链接】ref-extractor Reference Extractor - Extract Zotero/Mendeley references from Microsoft Word files 项目地址: https://gitcode.com/gh_mirrors/r…...

权限不是配置,是计算——MCP 2026动态分配核心算法解析,含PDP策略决策树与PEP响应延迟压测数据(实测<12ms)

更多请点击: https://intelliparadigm.com 第一章:权限不是配置,是计算——MCP 2026动态分配范式革命 在 MCP(Multi-Context Permissioning)2026 架构中,权限不再由静态策略文件或 RBAC 角色模板预定义&am…...

自动驾驶算法岗必备:手把手教你优化C++角度归一化代码(从Apollo源码说起)

自动驾驶算法岗必备:深度解析C角度归一化的工程实践与性能优化 在自动驾驶系统的开发中,角度归一化是一个看似简单却至关重要的基础操作。当车辆需要计算转向角度、航向偏差或传感器数据融合时,正确处理角度范围直接关系到算法的稳定性和可靠…...

手把手教你用VASP和p4vasp模拟STM图像:从DOS计算到PARCHG文件处理

从零开始掌握VASP与p4vasp的STM图像模拟全流程 在表面科学和材料研究领域,扫描隧道显微镜(STM)图像模拟已成为理论验证实验的重要手段。对于刚接触计算材料学的科研人员来说,掌握VASP结合p4vasp的STM模拟全流程,不仅能提升研究效率&#xff0…...

MCP 2026固件级漏洞修复全流程,含华为/思科/Juniper设备兼容性适配表(附厂商未发布的Beta补丁包)

更多请点击: https://intelliparadigm.com 第一章:MCP 2026固件级漏洞的原理与影响面深度解析 MCP 2026 是一款广泛应用于工业网关与边缘计算设备的微控制器协处理器,其固件中存在一个未经验证的 SMI(System Management Interrup…...

CLion远程调试踩坑实录:当GDBServer版本不匹配时,我们该如何优雅解决?

CLion远程调试实战:GDBServer版本冲突的终极解决方案 当你在嵌入式开发中满怀期待地启动CLion的远程调试功能,却在控制台看到"Protocol error"或"Unknown command"的红色报错时,那种挫败感每个开发者都深有体会。版本不匹…...

OpenCore配置终极指南:OCAuxiliaryTools图形化配置工具完全解析

OpenCore配置终极指南:OCAuxiliaryTools图形化配置工具完全解析 【免费下载链接】OCAuxiliaryTools Cross-platform GUI management tools for OpenCore(OCAT) 项目地址: https://gitcode.com/gh_mirrors/oc/OCAuxiliaryTools 你是否曾…...

Armv8-M安全扩展架构解析与实践指南

1. Armv8-M安全扩展架构解析Armv8-M安全扩展(Security Extension)为嵌入式系统提供了硬件级的安全隔离机制,其核心设计理念是通过划分安全(Secure)与非安全(Non-secure)状态来实现资源隔离。这种…...

终极JSXBIN解码指南:快速解密Adobe脚本加密格式的完整教程

终极JSXBIN解码指南:快速解密Adobe脚本加密格式的完整教程 【免费下载链接】jsxer A fast and accurate JSXBIN decompiler. 项目地址: https://gitcode.com/gh_mirrors/js/jsxer 在Adobe创意生态系统中,JSXBIN格式一直是开发者维护和审计脚本代码…...

抖音去水印工具终极指南:5分钟掌握TikTokDownload批量下载技巧

抖音去水印工具终极指南:5分钟掌握TikTokDownload批量下载技巧 【免费下载链接】TikTokDownload 抖音去水印批量下载用户主页作品、喜欢、收藏、图文、音频 项目地址: https://gitcode.com/gh_mirrors/ti/TikTokDownload 还在为抖音视频水印烦恼吗&#xff1…...

计算机生成全息术与JPEG压缩的融合优化

1. 计算机生成全息术与JPEG压缩的跨界融合在增强现实(AR)和虚拟现实(VR)近眼显示领域,计算机生成全息术(CGH)正逐渐成为实现真正三维显示的关键技术。与传统的立体显示不同,CGH通过数值模拟光衍射过程生成全息图,能够提供完整的视差信息和物理…...

终极解密:MS-DOS源代码如何塑造现代操作系统架构

终极解密:MS-DOS源代码如何塑造现代操作系统架构 【免费下载链接】MS-DOS The original sources of MS-DOS 1.25, 2.0, and 4.0 for reference purposes 项目地址: https://gitcode.com/GitHub_Trending/ms/MS-DOS MS-DOS作为个人计算机革命的开端&#xff0…...