QT笔记——QProcess学习

我们常常想通过某一个类,来启动一个外部进程

本文将讲解如何通过QProcess来进行启动外部进程

一:了解QProcess

QProcess是Qt框架提供的一个类,用于在应用程序中执行外部进程。它提供了一系列函数来启动、控制和与外部进程进行交互

1.启动进程的方式:

(1.1)分离式:外部程序启动程序后,主程序退出时,被调用的进程继续执行,不退出

cpp 复制代码
[static] bool QProcess::startDetached(const QString &program, const QStringList &arguments, const QString &workingDirectory = QString(), qint64 *pid = nullptr)

(1.2)一体式:当主程序退出时,被主程序调用起来的进程也退出

start还有其他的重载的函数,下面是其中一个

cpp 复制代码
void QProcess::start(const QString &program, const QStringList &arguments, QIODevice::OpenMode mode = ReadWrite)

2:常用的阻塞函数

cpp 复制代码
// 主进程阻塞,直到外部程序启动完毕,
waitForStarted()
 
// 主进程阻塞,直到外部程序执行完毕
waitForFinished() 

3:被调用进程接受数据

cpp 复制代码
QStringList list = QCoreApplication::arguments();
//为什么试3个呢,我明明参数是传的两个  因为第一个参数是我们传的exe启动 全路径 C:\Users\admin\Desktop\a.exe
if (list.size() == 3)
{
	QString exeFileName= list.at(0);
	QString oneStr = list.at(1);
	QString twoStr= list.at(2);
}

4:主进程接受被调用进程的数据

(4.1)使用QFile输出内容被 主进程捕获

cpp 复制代码
QFile file;
file.open(1, QFile::WriteOnly);
file.write("finished");
file.close();

(4.2)使用输出流 被 主进程捕获

cpp 复制代码
 cout << "it's message" << endl;

5:常用的信号

cpp 复制代码
//启动完毕
connect(process, &QProcess::started, this, [=]()
{
			
});

//捕获到消息时
connect(process, &QProcess::readyReadStandardOutput, this, [=]()
{

});

//完成
connect(process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished), this, [=](int exitCode, QProcess::ExitStatus exitStatus)
{

});

//状态改变
connect(process, &QProcess::stateChanged, this, [=](QProcess::ProcessState state)
{
	
});

二:使用QPrecess

1:startDetached 启动进程

1:使用了startDetached来启动进程,信号都是无效的,接受不到任何消息

2:当主进程关闭时,被调用的QTcpClientTest.exe 是不会退出的

3:主进程发送的参数,被调用的QTcpClientTest.exe 依然可以接受

cpp 复制代码
void QTcpSeverTest::on_btn1_clicked()
{
	QProcess* process = new QProcess(this);
	QString str = QApplication::applicationDirPath() + "/QTcpClientTest.exe";

	QStringList list;
	list.append("123");
	list.append("456");

	//分离式 启动外部进程
	process->startDetached(str, list);
	
	//无效
	connect(process, &QProcess::started, this, [=]()
		{
			qDebug() << "started";
		});

	//无效
	connect(process, &QProcess::readyReadStandardOutput, this, [=]()
		{
			QString qstr(process->readAllStandardOutput());
			qDebug() << "startDetached:" << qstr;
		});

	connect(process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished), this, [=](int exitCode, QProcess::ExitStatus exitStatus)
		{
			QString result = process->readAll();
			qDebug() << "result:" << result;
		});

	//无效
	connect(process, &QProcess::stateChanged, this, [=](QProcess::ProcessState state)
		{
			qDebug() << "show state:";
			switch (state)
			{
			case QProcess::NotRunning:
				qDebug() << "Not Running";
				break;
			case QProcess::Starting:
				qDebug() << "Starting";
				break;
			case QProcess::Running:
				qDebug() << "Running";
				break;
			default:
				qDebug() << "otherState";
				break;
			}

		});

}

2:start启动进程

1:使用了startDetached来启动进程,信号都是可以接受的

2:当主进程关闭时,被调用的QTcpClientTest.exe 是随着主进程一起退出

3:主进程发送的参数,被调用的QTcpClientTest.exe 可以接受参数

cpp 复制代码
	QProcess* process = new QProcess(this);
	QString str = QApplication::applicationDirPath() + "/QTcpClientTest.exe";

	QStringList list;
	list.append("bbbbb");
	list.append("aaaaa");

	//启动完毕
	connect(process, &QProcess::started, this, [=]()
	{
			qDebug() << "started:" ;
	});

	//捕获到消息时
	connect(process, &QProcess::readyReadStandardOutput, this, [=]()
	{
			QString qstr(process->readAllStandardOutput());
			qDebug() << "readyReadStandardOutput:" << qstr;
	});

	//完成
	connect(process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished), this, [=](int exitCode, QProcess::ExitStatus exitStatus)
	{
			if (exitStatus == QProcess::NormalExit) {
				qDebug() << "Process finished with exit code:" << exitCode;
			}
			else {
				qDebug() << "Process crashed!";
			}
	});

	//状态改变
	connect(process, &QProcess::stateChanged, this, [=](QProcess::ProcessState state)
	{
			qDebug() << "show state:";
			switch (state)
			{
			case QProcess::NotRunning:
				qDebug() << "Not Running";
				break;
			case QProcess::Starting:
				qDebug() << "Starting";
				break;
			case QProcess::Running:
				qDebug() << "Running";
				break;
			default:
				qDebug() << "otherState";
				break;
			}

	});

	//一体式  启动外部进程
	process->start(str, list);

3:execute()启动进程

1:使用了execute来启动进程,信号是无效的

2:主进程一直处于阻塞状态,等待被调用的QTcpClientTest.exe完成(关闭软件)

cpp 复制代码
	QProcess* process = new QProcess(this);
	QString str = QApplication::applicationDirPath() + "/QTcpClientTest.exe";

	QStringList list;
	list.append("bbbbb");
	list.append("aaaaa");

	//无效
	connect(process, &QProcess::started, this, [=]()
		{
			qDebug() << "started:";
		});

	//无效
	connect(process, &QProcess::readyReadStandardOutput, this, [=]()
		{
			QString qstr(process->readAllStandardOutput());
			qDebug() << "readyReadStandardOutput:" << qstr;
		});

	//无效
	connect(process, QOverload<int, QProcess::ExitStatus>::of(&QProcess::finished), this, [=](int exitCode, QProcess::ExitStatus exitStatus)
		{
			QString result = process->readAll();
			qDebug() << "result:" << result;
		});

	//无效
	connect(process, &QProcess::stateChanged, this, [=](QProcess::ProcessState state)
		{
			qDebug() << "show state:";
			switch (state)
			{
			case QProcess::NotRunning:
				qDebug() << "Not Running";
				break;
			case QProcess::Starting:
				qDebug() << "Starting";
				break;
			case QProcess::Running:
				qDebug() << "Running";
				break;
			default:
				qDebug() << "otherState";
				break;
			}

		});

	//如果进程 QTcpClientTest 不关闭 或者完成 , 则此进程一直卡住
	int exitCode = QProcess::execute(str, list);
	if (exitCode != 0) {
		qDebug() << "外部程序执行失败";
	}

	//接受进程捕获到输出的消息
	QByteArray output = process->readAllStandardOutput();
	QString msg = QString::fromLocal8Bit(output);
	qDebug() << msg;

打印的信息为:QIODevice::read (QProcess): device not open,说明execute没有打开设备,接受不到消息

4:start() + waitForFinished()启动进程

1:使用了execute来启动进程,信号是无效的

2:主进程一直处于阻塞状态,等待被调用的QTcpClientTest.exe完成(关闭软件)

cpp 复制代码
	QProcess* process = new QProcess(this);
	QString str = QApplication::applicationDirPath() + "/QTcpClientTest.exe";

	QStringList list;
	list.append("bbbbb");
	list.append("aaaaa");

	//启动进程
    process->start(str,list);

    // 等待进程完成
    process->waitForFinished();

	//接受进程捕获到输出的消息
    QByteArray output = process->readAllStandardOutput();
    QString msg = QString::fromLocal8Bit(output);
    qDebug() << msg;

完整代码:
QProcess完整学习代码

参考博客:
QProcess使用 一
QProcess使用 二
QProcess使用 三
QProcess使用 四
QProcess使用 五

相关推荐
eybk5 小时前
Pytorch+Mumu模拟器+萤石摄像头实现对小孩学习的监控
学习
6.945 小时前
Scala学习记录 递归调用 练习
开发语言·学习·scala
Aileen_0v06 小时前
【AI驱动的数据结构:包装类的艺术与科学】
linux·数据结构·人工智能·笔记·网络协议·tcp/ip·whisper
守护者1707 小时前
JAVA学习-练习试用Java实现“使用Arrays.toString方法将数组转换为字符串并打印出来”
java·学习
学会沉淀。7 小时前
Docker学习
java·开发语言·学习
Rinai_R7 小时前
计算机组成原理的学习笔记(7)-- 存储器·其二 容量扩展/多模块存储系统/外存/Cache/虚拟存储器
笔记·物联网·学习
吃着火锅x唱着歌7 小时前
PHP7内核剖析 学习笔记 第四章 内存管理(1)
android·笔记·学习
ragnwang7 小时前
C++ Eigen常见的高级用法 [学习笔记]
c++·笔记·学习
kiiila8 小时前
【Qt】对象树(生命周期管理)和字符集(cout打印乱码问题)
开发语言·qt
胡西风_foxww8 小时前
【es6复习笔记】rest参数(7)
前端·笔记·es6·参数·rest