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

从付费软件到自主开发:我用AI和FFmpeg实现了一个录屏工具棺

我为什么会发出这个疑问呢?是因为我研究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/632680/

相关文章:

  • 4090D显卡专属优化!Guohua Diffusion国风绘画工具部署教程
  • 解锁地理空间智能:用TorchGeo构建遥感深度学习应用
  • 金融中各类账户
  • “你用AI,那我也会用AI,我还要你干什么?”淹
  • 3.1 状态管理概述
  • Granite TimeSeries FlowState R1预测气象数据:温度与降水序列生成效果实录
  • 2026年非开挖顶管:管道堵塞非开挖疏通/管道塌陷非开挖修复/管道大堵头非开挖/管道气囊堵水非开挖/管道非开挖修复工艺/选择指南 - 优质品牌商家
  • 2026Q2钛合金门技术解析:断桥窗/钛合金门/钢质门/铝合金窗/防火窗/防火门/防爆门/防盗门/隔音门/不锈钢门/选择指南 - 优质品牌商家
  • CogVideoX-2b部署优势:相比云端API的成本效益对比
  • Open UI5 源代码解析之972:ValueStateHeader.js
  • Go语言中的反射与接口:从原理到实践
  • ASP.NET Core 外部依赖调用治理实战:HttpClientFactory、Polly 与幂等边界岩
  • Qwen-Image-2512实战教程:如何用API批量生成1000张不同尺寸像素头像
  • Agent Client Protocol 全景解析把
  • DAMA-DMBOK
  • RexUniNLU模型蒸馏实战:小模型保留大模型能力
  • 万物识别镜像新手入门:5分钟搭建你的第一个图像识别应用
  • ClawdBot实战体验:手把手教你搭建个人AI助手,效果惊艳
  • 从Market1501到实战:手把手教你用FastReID复现行人重识别SOTA模型(附避坑指南)
  • GTE语义搜索优化:提升企业文档检索准确率50%
  • 终极指南:使用over-golang构建分布式系统的etcd服务发现与gRPC集成方案
  • PP-DocLayoutV3入门必看:WebUI中彩色标注框颜色映射表(红/绿/紫/橙/黄)
  • [精品]基于微信小程序的基于企业微信的问卷系统的设计与实现 UniApp
  • FlowState Lab与MySQL联动:海量时空模拟数据的存储与检索方案
  • 云容笔谈·东方红颜影像生成系统Python入门实战:快速搭建AI绘画环境
  • 通俗易懂讲PIC单片机:从一窍不通到入门进步
  • 10个实用技巧:r2 HTTP客户端打造企业级请求的完整指南
  • 2026评价高的新型悬挑工字钢租赁企业盘点与采购参考:老式工字钢租赁、铁路钢板租赁、工地工字钢租赁、工地钢板租赁选择指南 - 优质品牌商家
  • 知壹网-中医资源库
  • GPUStack 在华为昇腾 I A 服务器上的保姆级部署指南举