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

小龙虾的自我养成之路

我为什么会发出这个疑问呢是因为我研究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 的 顺序消费 特性保持和底层系统的一致性。穆涛透崖

相关文章:

小龙虾的自我养成之路

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

免费3D重建神器Meshroom完全指南:从照片到专业模型的终极教程

免费3D重建神器Meshroom完全指南:从照片到专业模型的终极教程 【免费下载链接】Meshroom Node-based Visual Programming Toolbox 项目地址: https://gitcode.com/gh_mirrors/me/Meshroom 你是否梦想过将手机拍摄的普通照片变成逼真的3D模型?现在…...

智慧树自动刷课插件终极教程:3步实现高效学习自动化 [特殊字符]

智慧树自动刷课插件终极教程:3步实现高效学习自动化 🚀 【免费下载链接】zhihuishu 智慧树刷课插件,自动播放下一集、1.5倍速度、无声 项目地址: https://gitcode.com/gh_mirrors/zh/zhihuishu 还在为智慧树平台的繁琐操作而烦恼吗&am…...

VoxelNet论文精读与复现笔记:从体素划分到RPN,一步步拆解3D检测核心

VoxelNet论文精读与复现笔记:从体素划分到RPN,一步步拆解3D检测核心 在自动驾驶和机器人感知领域,3D目标检测一直是核心技术难题。传统方法依赖手工设计特征,而VoxelNet首次实现了从原始点云到3D边界框的端到端学习。本文将带您深…...

从map到base_link:深入解析ROS激光SLAM中的坐标变换链与数据流

1. 激光SLAM中的坐标系基础认知 第一次接触ROS激光SLAM时,我被各种坐标系搞得晕头转向。直到有次调试机器人导航时,发现地图总是偏移,才真正意识到坐标系理解的重要性。在激光SLAM系统中,数据就像接力赛跑,需要经过多个…...

Late:本地优先的编程智能体

如果能在 5GB 显存上使用本地 Qwen3.5-35B-A3B 编排代码库(通过 llama.cpp 达到约 25-30 tokens/sec,65k 上下文,其余层卸载到系统内存),你觉得如何? 更妙的是,两个并行的 agent 实例可以舒适地…...

高效使用NotebookLM的5种方法

如果你曾经被笔记、PDF、研究论文,甚至自己的想法弄得焦头烂额,那么谷歌的 NotebookLM 很可能会成为你最喜欢的新工具。 你可以把它想象成一个智能研究助手,它不仅真正理解你的文档,还能帮助你更好地思考,而不仅仅是更…...

Qianfan-OCR效果分享:培训教材PPT→知识点分级+案例引用+习题答案结构化

Qianfan-OCR效果分享:培训教材PPT→知识点分级案例引用习题答案结构化 1. 工具概览 Qianfan-OCR是基于百度千帆InternVL架构开发的单卡GPU专属文档解析工具,专为解决教育培训场景中的复杂文档解析需求而设计。与传统OCR工具不同,它不仅能识…...

使用FCM进行编码解码

文章目录1 FCM到底是什么?2 为什么论文里要用FCM?3 FCM输出的两个核心结果是什么?1. prototype / cluster centers2. membership matrix4 FCM到底在优化什么?5 FCM是怎么一步一步算出来的?第一步:先定簇数 c第二步:初始化隶属度矩阵第三步&#xff…...

TuShare的注册和使用

前言 TuShare是比较知名的证券第三方数据提供网站,最近我被试用各种爬虫搞烦了。打算花点钱直接试用第三方的数据看看。 1、TuShare上账号的注册 打开TuShare网址 https://tushare.pro/2、安装Tushare对应的包 Pip install tushare如果安装途中有断开的话&#xff0…...

在PyCharm的Django工程中修改初始页

1、原始的初始页2、setting.py中添加应用在quiz_site的setting.py 进行应用到的设置(对应的是一个根目录下的文件夹)3、Quiz_site\urls.py中的设置4、修改quiz/urls.py修改后的内容如下:5、views.py的设置如图:6、建立主页在quiz目…...

全栈编程基础知识8

全栈编程基础知识81.Redis1.介绍:基于C语言。开发的一款nosql数据库,基于内存的,读写快,支持多种数据类型。存的是键值对。2.类型:string list hash set bitmap,sorted set等。支持事务,lua脚本…...

大模型RAG (三)

一、文档的加载和分割1、文档LLM回复系统搭建2、把文本切分成chunks把文本切分成chunks的方式有很多种: 1.按照句子来切分 2.按照字符数来切分 3.按固定字符数结合overlapping window 4. 递归方法 RecursiveCharacterTextSplitter案例1: 按照句子来切分import retxt…...

航空特色学校建设实施方案

【核心亮点】1. "三维一体"课程体系:● 基础课(航空原理/飞行启蒙) 专项课(模拟飞行、航模制作、无人机编程) 学科融合课(航空物理/美术/编程),覆盖全学段。● 自主研发校…...

学工平台变革之旅:从管理到成长赋能,真正为学生点亮前行之路

✅作者简介:合肥自友科技 📌核心产品:智慧校园平台(包括教工管理、学工管理、教务管理、考务管理、后勤管理、德育管理、资产管理、公寓管理、实习管理、就业管理、离校管理、科研平台、档案管理、学生平台等26个子平台) 。公司所有人员均有多…...

智慧校园的权限管控,如何按角色精准设置操作范围?

✅作者简介:合肥自友科技 📌核心产品:智慧校园平台(包括教工管理、学工管理、教务管理、考务管理、后勤管理、德育管理、资产管理、公寓管理、实习管理、就业管理、离校管理、科研平台、档案管理、学生平台等26个子平台) 。公司所有人员均有多…...

Java 25虚拟线程性能断崖式跃迁:阿里云真实订单链路压测数据(RT从412ms→23ms,附全链路火焰图)

第一章:Java 25虚拟线程演进脉络与高并发架构新范式Java 25正式将虚拟线程(Virtual Threads)从预览特性转为标准特性,标志着JVM并发模型进入“轻量级线程即原语”时代。这一转变并非孤立演进,而是历经Project Loom多年…...

【毕设】城镇保障性住房管理系统

💟博主:程序员俊星:CSDN作者、博客专家、全栈领域优质创作者 💟专注于计算机毕业设计,大数据、深度学习、Java、小程序、python、安卓等技术领域 📲文章末尾获取源码数据库 🌈还有大家在毕设选题…...

【毕设】城市公园信息管理系统的设计与实现

💟博主:程序员俊星:CSDN作者、博客专家、全栈领域优质创作者 💟专注于计算机毕业设计,大数据、深度学习、Java、小程序、python、安卓等技术领域 📲文章末尾获取源码数据库 🌈还有大家在毕设选题…...

从dbus-send到busctl:手把手教你迁移到更现代的D-Bus调试工具链

从dbus-send到busctl:现代D-Bus调试工具链迁移实战指南 如果你曾经在Linux系统中与D-Bus打交道,那么对dbus-send这个老牌命令行工具一定不陌生。它就像一把瑞士军刀,虽然功能全面但用起来总有些笨拙——复杂的参数构造、晦涩的输出格式、缺乏…...

DevExpress GridControl单元格合并后无法编辑?一个属性帮你避开这个坑

DevExpress GridControl单元格合并实战:解决编辑冲突与高级应用 当我们在企业级应用开发中使用DevExpress的GridControl时,单元格合并是一个常见的可视化需求。想象一下这样的场景:你的财务系统需要展示客户交易记录,而同一个客户…...

别再只会用Canny了!深入对比Sobel、Prewitt、LoG:OpenCV边缘检测算法选型与避坑指南

边缘检测算法深度解析:从Sobel到Canny的工程实践指南 在计算机视觉领域,边缘检测是图像处理中最基础也最关键的步骤之一。许多开发者习惯性地将Canny算子作为默认选择,却忽略了其他算法在不同场景下的独特优势。本文将带您深入理解主流边缘检…...

我的模型总在测试集翻车?可能是数据增强的‘姿势’不对!聊聊那些年我们踩过的坑

模型测试集翻车?数据增强的六大陷阱与实战解决方案 当你满怀期待地将精心调参的模型投入测试集,却发现性能断崖式下跌——这种挫败感每个算法工程师都深有体会。上周团队里一位资深研究员盯着0.23的测试F1分数苦笑:"训练集明明98%准确率…...

MATLAB优化实战:从fminsearch到fmincon的工程问题求解

1. MATLAB优化工具箱入门:从实际问题到数学模型 第一次接触MATLAB优化工具箱时,我被它强大的功能震撼到了。记得当时正在做一个机械臂参数标定的项目,需要根据实验数据反推关节参数。这个问题本质上就是个典型的无约束优化问题,正…...

**链路追踪实战:用Go语言打造分布式系统的“心跳图谱”**在微服务架构日益普及的今天,一

链路追踪实战:用Go语言打造分布式系统的“心跳图谱” 在微服务架构日益普及的今天,一个请求可能跨越多个服务、几十个中间件甚至上百个节点。当问题出现时,传统的日志排查方式早已力不从心。这时,链路追踪(Tracing&am…...

第三章 低通滤波(LPF)

一 应用场景及公式当负载设备VCC需要的电压是1V,但是我们有12V的电压时,就需要使用电阻分压。问题:非理想环境中12V的电压会有波动(噪声)的,故而分出来1V电压也是有噪声的。1.1 容抗公式 :容抗&#xff08…...

PostgreSQL WITH 子句详解

PostgreSQL WITH 子句详解 引言 在数据库查询中,WITH 子句(也称为公用表表达式或 Common Table Expressions,简称 CTE)是一种强大的工具,它允许开发者将查询结果集作为子查询或临时表使用。WITH 子句在 PostgreSQL 中有…...

FPGA实战:手把手教你用Verilog实现有符号数的四舍五入(附完整代码与仿真)

FPGA实战:手把手教你用Verilog实现有符号数的四舍五入(附完整代码与仿真) 在数字信号处理领域,有符号数的四舍五入是一个看似简单却暗藏玄机的操作。许多初学者在处理负数时常常会遇到意想不到的结果,这是因为负数的四…...

工业级3D打印机季度出货回暖,入门级市场再创新高

当行业讨论从“技术突破”逐渐转向“规模落地”,全球3D打印市场正在经历一轮更深层次的结构调整。从TCT Asia 2026现场的热度变化,到厂商战略重心的转移,可以明显感受到:增长的引擎正在重塑,市场逻辑也在悄然变化。而C…...

基于鸿蒙Electron框架的物体碰撞效果测试应用开发详解

欢迎加入开源鸿蒙PC社区: https://harmonypc.csdn.net/ atomgit开源仓库地址: https://atomgit.com/feng8403000/game_Collisioneffect 示例效果 基于鸿蒙Electron框架的物体碰撞效果测试应用开发详解示例效果技术栈选择前端技术后端技术技术优势应用功…...