Qt QProcess详解

1.简介

QProcess提供了在 Qt 应用程序中启动外部程序的方法。通过QProcess,你可以启动一个进程,与它通信(发送输入和读取输出),检查它的状态,以及等待它完成。这个类在执行系统命令、运行其他程序或脚本时非常有用。

2.基本使用

  • 启动进程: 使用start()、execute()、或startDetached()方法来启动一个进程,后面讲解区别。
  • 读取输出:使用readAllStandardOutput()和 readAllStandardError()方法来读取进程的标准输出和标准错误输出。
  • 进程状态: 使用state()方法来检查进程的状态,例如是否正在运行、是否已结束等。
  • 等待进程完成: 使用 waitForFinished()方法等待进程结束。也可以使用 exitCode()和exitStatus()方法来检查进程的退出码和退出状态。
  • 终止进程:使用kill()和terminal()来终止进程。

3.信号和槽

QProcess提供了一些信号,可以在特定事件发生时发出,例如:

  • started: 进程开始执行时发出。
  • readyReadStandardOutput: 进程的标准输出有数据可读时发出。
  • readyReadStandardError: 进程的标准错误输出有数据可读时发出。
  • finished: 进程结束时发出。
QProcess process;
process.start("ls", QStringList() << "-l" << "/home/user");connect(&process, &QProcess::readyReadStandardOutput, [&]() {qDebug() << "Standard output:" << process.readAllStandardOutput();
});connect(&process, &QProcess::readyReadStandardError, [&]() {qDebug() << "Standard error:" << process.readAllStandardError();
});connect(&process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished),[=](int exitCode, QProcess::ExitStatus exitStatus)
{qDebug() << "Process finished with exit code" << exitCode<<" "<<exitStatus;
});

4.启动进程的区别

  • start()

    • start方法用于启动一个外部程序,它是异步的,意味着它会在启动进程后立即返回,不会阻塞调用线程。
    • 使用start,你可以与进程交互,例如读取输出、写入输入、等待进程完成等。
    • 可以指定进程的参数、环境变量,并且可以选择性地重定向标准输入输出。
    • start方法适用于需要与启动的进程进行通信或需要监控进程状态的情况。
  • execute()

    • execute方法也是用于启动一个外部程序,但它是以同步方式进行的,会阻塞调用线程直到外部程序执行完毕。
    • execute在内部调用start,然后等待进程结束,并返回进程的退出代码。
    • 由于它是同步的,execute不适合需要与进程交互或不想阻塞主线程的应用程序。
    • execute不提供重定向标准输入输出的选项,也不允许在进程运行期间执行其他任务。
    • execute的使用比较简单,因为它会自动等待进程结束,并且返回退出代码。
  • startDetached()

    • startDetached方法用于在独立的进程中启动一个外部程序,它与父进程完全分离,不与父进程的QProcess实例进行通信,(这意味着无法使用信号-槽通信)。
    • startDetached 适用于启动那些不需要与父进程交互的独立应用程序。
    • 使用 startDetached启动的进程有自己的控制台窗口(在 Windows 上),而 start和 execute启动的进程通常与父进程共享控制台。
    • startDetached不需要父进程等待子进程结束,子进程可以在父进程退出后继续运行。

5.示例

示例1:打开记事本程序。

源码:

说明:这里打开记事本程序,绑定了三个信号,使用start的方式打开,因为是异步的方式,所以父程序和记事本程序都可以操作。如果关闭了父程序,记事本程序会跟着关闭。

如果使用waitForFinished方法,就只能操作子程序,会在子程序结束后才能够操作父程序。

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QProcess>
#include <QDebug>// 创建 QProcess 对象
QProcess *process = nullptr;MainWindow::MainWindow(QWidget *parent) :QMainWindow(parent),ui(new Ui::MainWindow)
{ui->setupUi(this);process = new QProcess(this);connect(process, &QProcess::readyReadStandardOutput, [&]() {qDebug() << "Standard output:" << process->readAllStandardOutput();});connect(process, &QProcess::readyReadStandardError, [&]() {qDebug() << "Standard error:" << process->readAllStandardError();});connect(process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished),[=](int exitCode, QProcess::ExitStatus exitStatus){qDebug() << "Process finished with exit code" << exitCode<<" "<<exitStatus;});
}MainWindow::~MainWindow()
{delete ui;
}void MainWindow::on_btnOpen_clicked()
{// 设置要打开的文件路径QString filePath = "C:/Users/Lenovo/Desktop/1.txt";// 启动记事本并打开文件process->start("notepad", QStringList() << filePath);// 如果需要,可以等待记事本关闭//process->waitForFinished();
}void MainWindow::on_btnClose_clicked()
{process->terminate();
}

关闭记事本程序后,会触发结束信号。

假如我们使用startDetached的方法打开记事本程序,因为它与父进程完全分离,不与父进程的QProcess实例进行通信,所以打开之后,我们无法再通过点击“关闭记事本”按钮关闭它,也无法接收到进程退出信号。如果关闭了父程序,记事本程序不会受到影响。

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QProcess>
#include <QDebug>// 创建 QProcess 对象
QProcess *process = nullptr;MainWindow::MainWindow(QWidget *parent) :QMainWindow(parent),ui(new Ui::MainWindow)
{ui->setupUi(this);process = new QProcess(this);connect(process, &QProcess::readyReadStandardOutput, [&]() {qDebug() << "Standard output:" << process->readAllStandardOutput();});connect(process, &QProcess::readyReadStandardError, [&]() {qDebug() << "Standard error:" << process->readAllStandardError();});connect(process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished),[=](int exitCode, QProcess::ExitStatus exitStatus){qDebug() << "Process finished with exit code" << exitCode<<" "<<exitStatus;});
}MainWindow::~MainWindow()
{delete ui;
}void MainWindow::on_btnOpen_clicked()
{// 设置要打开的文件路径QString filePath = "C:/Users/Lenovo/Desktop/1.txt";// 启动记事本并打开文件//process->start("notepad", QStringList() << filePath);process->startDetached("notepad", QStringList() << filePath);// 如果需要,可以等待记事本关闭//process->waitForFinished();
}void MainWindow::on_btnClose_clicked()
{process->terminate();
}
示例2:执行不带管道的linux命令。
#include "widget.h"
#include "ui_widget.h"
#include <QProcess>
#include <QDebug>// 创建 QProcess 对象
QProcess *process = nullptr;Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);process = new QProcess(this);connect(process, &QProcess::readyReadStandardOutput, [&]() {qDebug() << "Standard output:" << QString::fromUtf8(process->readAllStandardOutput());});connect(process, &QProcess::readyReadStandardError, [&]() {qDebug() << "Standard error:" << process->readAllStandardError();});connect(process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished),[=](int exitCode, QProcess::ExitStatus exitStatus){qDebug() << "Process finished with exit code" << exitCode<<" "<<exitStatus;});QStringList arguments;arguments << "-al" << "/home/zz";process->start("ls",arguments);process->waitForFinished();
}Widget::~Widget()
{delete ui;
}

运行结果: 

示例3:执行带有管道( | )的linux命令

process->start("bash", QStringList() << "-c" << "ps -ef | grep firefox");

QProcess不直接支持执行带有管道的命令,因为管道是shell的一个特性,而QProcess是直接执行程序而不是通过shell,可以尝试将整个管道命令作为单个字符串传递参数。

#include "widget.h"
#include "ui_widget.h"
#include <QProcess>
#include <QDebug>// 创建 QProcess 对象
QProcess *process = nullptr;Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);process = new QProcess(this);connect(process, &QProcess::readyReadStandardOutput, [&]() {qDebug() << "Standard output:" << QString::fromLocal8Bit(process->readAllStandardOutput());});connect(process, &QProcess::readyReadStandardError, [&]() {qDebug() << "Standard error:" << QString::fromLocal8Bit(process->readAllStandardError());});connect(process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished),[=](int exitCode, QProcess::ExitStatus exitStatus){qDebug() << "Process finished with exit code" << exitCode<<" "<<exitStatus;});process->start("bash", QStringList() << "-c" << "ps -ef | grep firefox");process->waitForFinished();
}Widget::~Widget()
{delete ui;
}

运行结果:

示例4:执行带有重定向( >> )的命令。

process->start("bash", QStringList() << "-c" << "ping 127.0.0.1 >> ./test.txt");

重定向到本地test.txt路径下。

#include "widget.h"
#include "ui_widget.h"
#include <QProcess>
#include <QDebug>// 创建 QProcess 对象
QProcess *process = nullptr;Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);process = new QProcess(this);connect(process, &QProcess::readyReadStandardOutput, [&]() {qDebug() << "Standard output:" << QString::fromLocal8Bit(process->readAllStandardOutput());});connect(process, &QProcess::readyReadStandardError, [&]() {qDebug() << "Standard error:" << QString::fromLocal8Bit(process->readAllStandardError());});connect(process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished),[=](int exitCode, QProcess::ExitStatus exitStatus){qDebug() << "Process finished with exit code" << exitCode<<" "<<exitStatus;});process->start("bash", QStringList() << "-c" << "ping 127.0.0.1 >> ./test.txt");process->waitForFinished();
}Widget::~Widget()
{delete ui;
}

运行结果:

示例5:传递子程序main函数参数。

假如两个程序都是自己开发的,父进程需要传递给子进程main函数参数,很简单

process->start("./parent.exe p1 p2 p3");

6.更多参考

Qt+FFmpeg+opengl从零制作视频播放器-1.项目介绍_qt opengl视频播放器-CSDN博客

libVLC 提取视频帧使用OpenGL渲染-CSDN博客

QT UDP通信(单播、广播、组播)-CSDN博客

QCharts -1.概述-CSDN博客

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

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

相关文章

AI大模型探索之路-应用篇11:AI大模型应用智能评估(Ragas)

目录 前言 一、为什么要做智能评估&#xff1f; 二、Ragas是什么&#xff1f; 三、Ragas使用场景 四、Ragas评估指标 五、Ragas代码实践 总结 前言 随着人工智能技术的飞速发展&#xff0c;AI大模型&#xff08;LLM&#xff09;已经成为了推动技术创新和应用的关键因素。…

深入理解信号上升沿与带宽的关系

信号的上升时间&#xff0c;对于理解信号完整性问题至关重要&#xff0c;高速pcb设计中的绝大多数问题都和它有关&#xff0c;很多信号完整性问题都是由信号上升时间短引起的&#xff0c;你必须对他足够重视。 信号上升时间并不是信号从低电平上升到高电平所经历的时间&#xf…

连接两部VR头显的type-c DP分配器方案,可以给主机设备PD反向供电与两部VR同时供电。

随着type-c的发展&#xff0c;目前越来越多的设备都在使用type-c作为连接的接口&#xff0c; 不仅是笔记本与手机在使用现在的游戏主机如&#xff08;任天堂&#xff0c;steam&#xff0c;&#xff09;或者是VR的一体机或者是VR头显也都在使用type-c作为连接接口。 type-c接口…

传统图机器学习的特征工程-全图

将整张图表示成为一个低维向量&#xff0c;反映全图的特征 key idea&#xff1a;Bag-of-Words&#xff08;BOW&#xff09;把图看作文章&#xff0c;把节点看作单词 Kernel mothods

FL Studio808鼓音在哪 FL Studio怎么让音乐鼓点更有力 FL Studio教程

FL Studio808鼓音在哪&#xff1f;808是一款电鼓机的名称&#xff0c;它发出的声音也被称之为808鼓&#xff0c;通常我们可以安装鼓机插件来使用&#xff0c;但FL Studio中自带的也有808鼓的采样音频。FL Studio怎么让音乐鼓点更有力&#xff1f;让鼓点更有力要从EQ均衡器、压缩…

ELK日志分析系统(上)

目录 引言 一、ELK日志分析系统简介 1.1 日志服务器 1.2 ELK日志分析系统的组成 1.3 日志处理步骤 二、Elasticsearch介绍 2.1 概述 2.2 核心概念 三、Logstash介绍 3.1 概述 3.2 主要组件 四、Kibana介绍 4.1 概述 4.2 主要功能 五、ELK的工作原理 六、部署ELK…

LINUX系统编程:缓冲区

1.为什么要有缓冲区 缓冲区分成语言层面的缓冲区和操作系统层面的缓冲区 先说结论&#xff0c;语言的缓冲区可以减少系统调用的次数进而提高向文件写入和读取的效率。 2.举例子 向屏幕打印&#xff0c;无非就是向屏幕这个文件的缓冲区写入&#xff0c;然后在由操作系统刷新…

js学习总结

这里写目录标题 前情提要JavaScript书写位置1. 内部javaScript (不常用)2. 外部javaScript (常用)3.内联javaScript (常用) js中的输入和输出输出语法1. document.write()2. alert()3. console.log() 输入语法prompt() 前情提要 1. 在javaScript中的 分号 是可以省略的JavaScr…

车机系统与 Android 的关系概述

前言&#xff1a;搞懂 Android 系统和汽车到底有什么关系。 文章目录 一、基本概念1、Android Auto1&#xff09;是什么2&#xff09;功能 2、Google Assistant3、Android Automotive1、Android Auto 和 Android Automotive 的区别 4、App1&#xff09;App 的开发2&#xff09;…

亚马逊云科技CTO带你学习云计算降本增效秘诀

2023亚马逊云科技一年一度的重磅春晚--Re:invent上有诸多不同话题的主题Keynote&#xff0c;这次小李哥带大家复盘来自亚马逊CTO: Wener博士的主题演讲: 云架构节俭之道1️⃣节俭对于云计算为什么重要&#xff1f; ▶️企业基础设施投入大&#xff0c;利用好降本策略可以减少巨…

个人笔记目录

目录 一、lora 微调 alpaca 笔记 二、全量微调 Llama2-7b笔记 三、Huggingface trainer 与 from_pretrained简单介绍&#xff08;笔记&#xff09; 四、vscode调试launch.json常用格式 五、huggingface generate函数简介 六、Trl: llama2-7b-hf使用QLora 4bit量化后ds zer…

Java项目-源码!大学生兼职信息系统

大学生兼职信息系统 1、功能介绍1.1、演示视频 2、系统部分功能展示2.1、管理员登录2.2、管理员功能模块2.2.1、轮播图管理2.2.2、招聘信息管理2.2.3、企业信息管理 3、系统概述4、开发环境 1、功能介绍 本文以Java为开发技术&#xff0c;实现了一个大学生兼职信息系统。 功能…

RestTemplate详解

一、概述 文章参考Spring之RestTemplate详解 1.1介绍 现如今的 IT 项目&#xff0c;由服务端向外发起网络请求的场景&#xff0c;基本上处处可见&#xff01; 传统情况下&#xff0c;在服务端代码里访问 http 服务时&#xff0c;一般会使用 JDK 的 HttpURLConnection 或者 A…

kali工具----网络映射器(Network Mapper)系统指纹

系统指纹识别 现在一些便携式计算机操作系统使用指纹识别来验证密码进行登录。指纹识别是识别系统的一个典型模式&#xff0c;包括指纹图像获取、处理、特征提取和对等模块。如果要做渗透测试&#xff0c;需要了解要渗透测试的操作系统的类型才可以。本节将介绍使用Nmap工具测试…

【OTA】STM32-OTA升级——持续更新

【OTA】STM32-OTA升级——持续更新 文章目录 前言一、ymodem串口协议1、Ymodem 协议2、PC3、蓝牙4、WIFI云平台 二、UDS车载协议1.UDS协议 总结 前言 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考 一、ymodem串口协议 1、Ymodem 协议 STM32 Ymodem …

【第三十篇】并发插件 turbo intruder 安装及使用教程

文章目录 安装使用本篇主要介绍turbo intruder的两种安装方式及使用教程。 安装 1、在BurpSuite的扩展模块的BAPP商店中找到turbo intruder,点击安装即可使用: 2、若在BurpSuite中无法直接安装,可手动添加该插件。 具体操作: 进入Github中安装: https://github.com/Po…

代码随想录算法训练营三刷day53 | 动态规划之子序列 1143.最长公共子序列 1035.不相交的线 53. 最大子序和

day53 1143.最长公共子序列1.确定dp数组&#xff08;dp table&#xff09;以及下标的含义2.确定递推公式3.dp数组如何初始化4.确定遍历顺序5.举例推导dp数组 1035.不相交的线53. 最大子序和1.确定dp数组&#xff08;dp table&#xff09;以及下标的含义2.确定递推公式3.dp数组如…

FireProx:一款功能强大的AWS API网关管理与IP地址轮换代理工具

关于FireProx FireProx是一款功能强大的AWS API网关安全管理工具&#xff0c;该工具可以帮助广大研究人员创建实现唯一IP地址轮换的实时HTTP转发代理。 在发送网络请求或进行网络交互时&#xff0c;实现源IP地址轮换是一个非常复杂的过程&#xff0c;虽然社区中也有相关的工具…

【STL详解 —— stack和queue的介绍及使用】

STL详解 —— stack和queue的介绍及使用 stackstack的定义方式stack的使用 queuequeue的定义方式queue的使用 stack stack是一种容器适配器&#xff0c;专门用在具有后进先出操作的上下文环境中&#xff0c;其只能从容器的一端进行元素的插入与提取操作。 stack的定义方式 首…

前端三剑客 —— JavaScript (第六节)

目录 内容回顾 BOM编程 DOM编程* document对象 document对象的属性 document对象的方法 DOM对象节点 操作DOM对象内容 操作DOM对象的属性 --- DOM对象.属性名称 --- DOM对象[属性名称] --- 调用系统API &#xff08;Application Program interface&#xff09;&#…