破解 Qt QProcess 在 Release 模式下的“卡死”之谜

在使用 Qt 的 QProcess 以调用外部 ffmpeg/ffprobe 进行音视频处理时,常见的工作流程是:

  1. gatherParams:通过 ffprobe 同步获取媒体文件的参数(分辨率、采样率、声道数、码率等)。

  2. reencode:逐个文件调用 ffmpeg -crf 或者 ffmpeg -c:a libmp3lame,异步重新编码到统一格式。

  3. concat:生成 concat_list.txt 后,调用 ffmpeg -f concat 将中间文件拼接成最终输出。

Debug 模式下一切运行正常,但切换到 Release(尤其是打包到生产环境)后,却经常出现程序卡在“开始拼接…”或无法结束 waitForFinished(),不抛错也不返回的怪异现象。本文将深入剖析其背后的根因,并提供简单可靠的解决方案。


一、问题重现

  • 视频合并VideoMergerstartConcat() 调用 ffmpeg -f concat 后,永远等不到 finished(),进度条卡住。

  • 音频合并:同理,AudioMerger 在拼接阶段也陷入“死循环”。

  • Debug 模式:IDE 输出面板能看到 ffmpeg 日志,流程正常结束。

  • Release 模式:IDE 不在;也没有 QProcess 输出日志;程序停在那不动。


二、为什么会卡死?

操作系统对每个子进程的 stdoutstderr 都设有管道缓冲区(通常约 4–64 KB 不等)。当 ffmpeg/ffprobe 输出日志到 stderr(ffmpeg 默认把进度和警告都输出到 stderr)时:

  1. 如果 没人读取,缓冲区一旦写满,子进程再写就被阻塞。

  2. 被阻塞的子进程卡在写日志上,无论它是否执行到“结束逻辑”,都无法调用 exit(),因此 Qt 收不到 finished() 信号。

  3. 程序就好像“挂住”了:既不成功也不报错。


三、为什么 Debug 下不出问题?

  1. IDE 帮你读管道
    在 Qt Creator、Visual Studio 等调试器中,子进程的 stderr/stdout 会被自动转发到“应用输出”视图——等于在后台不断做 read()

  2. 执行速度慢
    Debug 编译跑得更慢,ffmpeg 输出日志的速度往往跟不上缓冲区填满的节奏;另外,IDE 读缓冲区的效率也帮忙拉低了写入速度。


四、两个简洁可靠的解决方案

1. 持续 drain 输出
  • 优点:不修改 ffmpeg 参数,兼容所有场景。

  • 做法

    // 对于每个 QProcess,都这样设置并连接:
    proc->setProcessChannelMode(QProcess::MergedChannels);
    connect(proc, &QProcess::readyReadStandardError, this, [=](){proc->readAllStandardError(); // 或者 readAllStandardOutput()
    });
    
  • 原理:将 stdout 和 stderr 合并到一条管道,只需持续读取一次就能清空所有日志,避免缓冲区写满。

2. 直接转发到父进程
  • 优点:无需在代码里手动读;可以在控制台直接看到 ffmpeg 输出。

  • 做法

  • proc->setProcessChannelMode(QProcess::ForwardedChannels);
    

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/79574.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL 中 UPDATE 结合 SELECT 和 UPDATE CASE WHEN 的示例

概述 以下是 MySQL 中 UPDATE 结合 SELECT 和 UPDATE CASE WHEN 的示例: 一、UPDATE 结合 SELECT(跨表更新) 场景:根据 orders 表中的订单总金额,更新 users 表中用户的 total_spent 字段。 -- 创建测试表 CREATE T…

【MCP】魔搭社区MCP服务(高德地图、everything文件搜索)

【MCP】魔搭社区MCP服务(高德地图、everything文件搜索) 1、上手使用2、环境配置(1)cherry-studio配置(2)添加魔搭大模型服务(如果已经设置了其他大模型服务,可跳过)&…

MapReduce 的工作原理

MapReduce 是一种分布式计算框架,用于处理和生成大规模数据集。它将任务分为两个主要阶段:Map 阶段和 Reduce 阶段。开发人员可以使用存储在 HDFS 中的数据,编写 Hadoop 的 MapReduce 任务,从而实现并行处理1。 MapReduce 的工作…

MCU开启浮点计算FPU

FPU 测试 1. FPU 简介2. 协处理器控制寄存器(CPACR)3. 开启FPU4. 验证FPU(Julia 分形实验) 1. FPU 简介 FPU 即浮点运算单元(Float Point Unit)。浮点运算,对于定点 CPU(没有 FPU 的…

进程相关面试题20道

一、基础概念与原理 1.进程的定义及其与程序的本质区别是什么? 答案:进程是操作系统分配资源的基本单位,是程序在数据集合上的一次动态执行过程。核心区别:​ 动态性:程序是静态文件,进程是动态执行实例…

React Hooks 精要:从入门到精通的进阶之路

Hooks 是 React 16.8 引入的革命性特性,它让函数组件拥有了类组件的能力。以下是 React Hooks 的详细使用指南。 一、基础 Hooks 1. useState - 状态管理 import { useState } from react;function Counter() {const [count, setCount] = useState(0); // 初始值为0return …

springboot3+vue3融合项目实战-大事件文章管理系统-更新用户头像

大致分为三步 首先在usercontroller里面加入方法 PatchMapping ("/updateAvatar")public Result upadateAvatar(RequestParam URL String avatarUrl){userService.updateAvater(avatarUrl);return Result.success();}url注解能验证传入的url是不是合法的&#xff0c…

Mosaic数据增强技术

Mosaic 数据增强技术是一种在计算机视觉领域广泛应用的数据增强方法。下面是Mosaic 数据增强技术原理的详细介绍 一、原理 Mosaic 数据增强是将多张图像(通常是 4 张)按照一定的规则拼接在一起,形成一张新的图像。在拼接过程中,会…

Git安装教程及常用命令

1. 安装 Git Bash 下载 Git 安装包 首先,访问 Git 官方网站 下载适用于 Windows 的 Git 安装包。 安装步骤 启动安装程序:双击下载的 .exe 文件,启动安装程序。选择安装选项: 安装路径:可以选择默认路径&#xff0…

学习日志04 java

PTA上的练习复盘 java01 编程题作业感悟: 可以用ai指导自己怎么调试,但是不要把调代码这过程里面的精华交给ai,就是自己去修正错误不能让ai代劳!~~~ 1 scanner.close() Scanner *** new Scanner(System.in); ***.close(); …

AI 在模仿历史语言方面面临挑战:大型语言模型在生成历史风格文本时的困境与研究进展

概述 在当今数字化时代,人工智能(AI)技术在诸多领域展现出了强大的能力,但在处理历史语言这一特定任务时,却遭遇了不小的挑战。美国和加拿大的研究人员通过合作发现,像 ChatGPT 这样的大型语言模型&#x…

基于 Spring Boot 瑞吉外卖系统开发(十二)

基于 Spring Boot 瑞吉外卖系统开发(十二) 菜品删除 单击“批量删除”和“删除”时,会携带需要删除的菜品的id以delete请求方式向“/dish”发送请求。 URLhttp://127.0.0.1:8080/dish调用方法DELETE参数ids DishController添加删除方法 …

Day22打卡-复习

复习日 仔细回顾一下之前21天的内容,没跟上进度的同学补一下进度。 作业: 自行学习参考如何使用kaggle平台,写下使用注意点,并对下述比赛提交代码 泰坦尼克号人员生还预测https://www.kaggle.com/competitions/titanic/overview K…

L48.【LeetCode题解】904. 水果成篮

目录 1.题目 2.分析 方法1:暴力枚举 方法2:暴力解法的优化:滑动窗口 代码 方法3:优化方法2:使用数组充当哈希表 方法4:四个变量分别充当篮子和篮子中水果的个数(最快!!!) 代码 容易忽略的点 1.题目 https://leetcode.cn/problems/fruit-into-baskets/ 你正在探访一家农…

Leetcode-BFS问题

LeetCode-BFS问题 1.Floodfill问题 1.图像渲染问题 [https://leetcode.cn/problems/flood-fill/description/](https://leetcode.cn/problems/flood-fill/description/) class Solution {public int[][] floodFill(int[][] image, int sr, int sc, int color) {//可以借助另一…

Typora+PicGo+Gitee图床配置教程 自动图片上传

配置步骤 #mermaid-svg-aPUbWs43XR5Rh7vf {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-aPUbWs43XR5Rh7vf .error-icon{fill:#552222;}#mermaid-svg-aPUbWs43XR5Rh7vf .error-text{fill:#552222;stroke:#552222;}#…

养生:开启健康生活的全新篇章

养生是一场关乎生活品质与身心健康的持续修行,从饮食调养到运动锻炼,从睡眠管理到心态塑造,每个环节都对健康有着深远影响。以下为你提供全面且实用的养生指南。 饮食养生:科学膳食,滋养生命 合理的饮食是养生的根基…

Python | 赤道频散关系图

写在前面 写开题报告, 想用个图发现截出来全是糊的。索性自己画了,主要实现的Matsuno(1966)的赤道波动频散关系图。但是,实在是没有审美,其他文献里都是黑色,这里非要用个紫色,因为…

Nexus 私有仓库 + Nginx 反向代理部署文档

1. 使用 Podman 部署 Nexus 3 podman run --name nexus -d \-p 8081:8081 \-v /data:/nexus-data \-v /etc/localtime:/etc/localtime \-e TZ"Asia/Shanghai" \-e INSTALL4J_ADD_VM_PARAMS"-Xms10240m -Xmx10240m -XX:MaxDirectMemorySize4096m" \docker.…

一.Gitee基本操作

一.初始化 1.git init初始化仓库 git init 用于在当前目录下初始化一个本地 Git 仓库,让这个目录开始被 Git 跟踪和管理。 生成 .git 元数据目录,从而可以开始进行提交、回退、分支管理等操作。 2.git config user.name/user.email配置本地仓库 # 设置…