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

相关文章:

  • 2026年出海泰国选哪家服务商?泰国名义雇主EOR公司推荐Safeguard Global - 品牌2026
  • 3分钟解放双手:Midscene让AI帮你完成所有浏览器重复操作
  • 北京家长亲测5家零差评美国留学中介,附详细对比测评! - 资讯焦点
  • 3步解锁Axure RP中文界面:告别英文菜单,提升原型设计效率75%
  • GKD订阅管理革命:一站式订阅库如何简化你的自动化体验
  • sysinfo 安全部署指南:在 macOS/iOS 沙盒环境中的正确使用方法
  • 避坑指南:OpenSIPS 3.1在Docker下的5个常见配置错误(附NAT解决方案)
  • 深度揭秘:如何用CDecrypt工具轻松解密Wii U游戏文件
  • 2026哪种净水器适合家用?不同家庭的适配方案,小白也能直接抄作业 - 资讯焦点
  • 盘点2026年莆田实力强的灯光设计品牌企业,费用多少 - mypinpai
  • RePKG:深度解析Wallpaper Engine资源提取与纹理转换的终极方案
  • 如何快速部署QQ截图独立版:完整配置与OCR优化指南
  • LVGL v8.3在GD32F450上跑飞?手把手教你用Keil调试HardFault_Handler定位真凶
  • 2026年4月一体化污水泵站厂家TOP3相关解析 - 资讯焦点
  • 2026年口碑好的CRM系统盘点:企业选型必看清单 - 纷享销客智能型CRM
  • 大模型面试通关秘籍:小白程序员必备的收藏级实战指南
  • 终极指南:10分钟掌握Bypass Paywalls Chrome Clean内容解锁神器
  • 找靠谱的电脑组装线定制厂家?这3个核心标准帮你避坑 - 丁华林智能制造
  • 三步搞定双语歌词:LrcHelper终极指南让音乐学习更简单
  • Kruskal-Wallis检验避坑指南:当你的数据不满足正态性假设时该怎么办?
  • 深度解析glogg:5个步骤掌握专业级跨平台日志分析工具实战指南
  • 城通网盘直连解析终极方案:如何让下载效率提升300%的完整开源工具
  • Qwen3-TTS-12Hz-1.7B-Base部署教程:国产昇腾/寒武纪AI芯片适配方案
  • 2026年出海欧洲选哪家服务商?欧洲名义雇主EOR公司推荐Safeguard Global - 品牌2026
  • ComfyUI-SeedVR2-Kontext实战:如何用RTX4090一键修复老照片(附完整工作流)
  • WorkshopDL完全教程:无需Steam客户端轻松下载1000+游戏模组
  • 2026年精选推荐,城投、商业、多业态资产管理系统选型指南 - 品牌2026
  • “闭眼入”的女鞋品牌是怎么炼成的?从品控到口碑 - 博客湾
  • 2026年福州靠谱的灯光设计服务公司排名,权威灯光设计推荐 - 工业品牌热点
  • 3分钟解锁付费内容:智能内容访问工具终极指南