Qt QProcess 进程间通信读写数据通信

本文介绍了如何使用Qt的QProcess  进行程序开发,包括启动进程间通信、设置环境变量、通用方法;方便在日常开发中使用;

1.使用Qt进行程序开发,可以通过QProcess类用于启动外部程序并与其进行通信.;

进程A(例如主程序)创建了一个QProcess B,这个B就称为A的子进程,而A称为B的父进程。

1.1. 运行进程 

可以使用 Start  or Open

要启动进程,需要运行的程序的名称和命令行参数作为参数传递给start()。参数以QStringList形式提供。
start()方法原型:

void start(const QString &program, const QStringList &arguments, OpenMode mode = ReadWrite)
void start(const QString &command, OpenMode mode = ReadWrite)

也可以使用setProgram()setArguments()设置要运行的程序,然后调用start()open()
以下是setProgram()setArguments()open()函数原型:

bool open(OpenMode mode = ReadWrite) Q_DECL_OVERRIDE;QString program() const;void setProgram(const QString &program);QStringList arguments() const;void setArguments(const QStringList & arguments);

1.2 环境运行设置

因为有一些程序会有依赖,所以需要设备运行环境:

通过调用setProcessEnvironment()为进程设置环境变量。

要设置工作目录,请调用setWorkingDirectory()。默认情况下,进程在调用进程的当前工作目录中运行

原型:

void QProcess::setProcessEnvironment(const QProcessEnvironment &environment)

样例:

  QProcess process;QProcessEnvironment env = QProcessEnvironment::systemEnvironment();env.insert("PATH", "E:\\Program"); // Add an environment variableprocess.setProcessEnvironment(env);process.start("app.exe");

1.3. 进程间通信
1.3.1关于父进程读写子进程的数据,主要是用到

子进程接收到了父进程数据,两个信号会发射出来:
void readyReadStandardError()
void readyReadStandardOutput()

父进程通过上面这两个信号,判断读取子进程发来的消息;

QByteArray QProcess::readAllStandardError()
QByteArray QProcess::readAllStandardOutput()

QProcess::write()发出信息

1.3.2 关于子进程读写父进程的数据,主要是用到

Windows中:需要开启一个线程来管理stdin的文件变化,这个需要使用Windows API函数
linux中:使用QSocketNotifier 监听 stdin文件,当改文件有变化是,读取信息

 

ReadFile(hStdinDup,chbuf,sizeof(chbuf),&dwRead,NULL);// get  hstdinDup handle data

这个是阻塞函数,类似控制台程序中的cin >> data,或者gets()。

因此需要开启一个线程来管理stdin的文件变化(函数中有涉及到while阻塞,一直检测标准输入通道stdin是否有可读取的信息,因此另开线程),在pro文件中添加:QT +=concurrent。实例代码如下:

QFuture<void> fu=QtConcurrent::run(this,&Widget::readStdin);// open a threadvoid Widget::readStdin()
{bool ok=true;char chbuf[1024]; DWORD dwRead; HANDLE hStdinDup;//HANDLE  const HANDLE hStdin=GetStdHandle(STD_INPUT_HANDLE);//GetStdHandle if(hStdin==INVALID_HANDLE_VALUE)// return;DuplicateHandle(GetCurrentProcess(),hStdin,GetCurrentProcess(),&hStdinDup,0,false,DUPLICATE_SAME_ACCESS); CloseHandle(hStdin);while(ok){ok=ReadFile(hStdinDup,chbuf,sizeof(chbuf),&dwRead,NULL);//emit sig_log(QLatin1String("ok is:")+QString::number(ok));if(ok &&dwRead!=0){emit sig_receivedCommand(QString::fromUtf8(chbuf,dwRead));}}}


 linux平台:QFile来读取标准输入

QProcess子进程通过QFile来读取标准输入来接收父进程信息。通过QFile绑定QSocketNotifier来接收标准输入的实时信号,因为QSocketNotifier的activated信号可以通过标准输入是否有消息实时触发。

QProcess子进程通过QFile绑定标准输出stdout来发送消息给父进程。
注意:子进程读取信息不能通过QFile的readline等接口读取父进程信息,因为QFile他会读取标准输入的所有信息,并且不到长度就没有返回
 

m_file.open(stdin, QFile::ReadOnly); //stdin=0if (m_file.isOpen()){m_clientsocket = new QSocketNotifier(m_file.handle(), QSocketNotifier::Read,this);connect(m_clientsocket,SIGNAL(activated(int)),this,SLOT(slotreaddata(int)));}void MainWindow::slotreaddata(int fd){if(fd != m_file.handle() )return;    char buf[128] = {0};read(fd,buf,sizeof(buf));      m_readedit->append(QString::fromUtf8("rev msg:%1").arg(buf));
}

2. 程序样例:

主进程
 

#include "mainwindow.h"
#include "ui_mainwindow.h"#include <QDebug>MainWindow::MainWindow(QWidget *parent) :QMainWindow(parent),ui(new Ui::MainWindow)
{ui->setupUi(this);
}MainWindow::~MainWindow()
{if(m_pProcess){m_pProcess->close();}delete ui;
}void MainWindow::on_btn_invokingClient_clicked()
{if(!m_pProcess){m_pProcess = new QProcess(this);// 完成时调用connect(m_pProcess, static_cast<void (QProcess::*)(int, QProcess::ExitStatus)>(&QProcess::finished), this, [=](int exitCode, QProcess::ExitStatus exitStatus){Q_UNUSED(exitCode)Q_UNUSED(exitStatus)});// 进程错误时触发connect(m_pProcess, static_cast<void (QProcess::*)(QProcess::ProcessError)>(&QProcess::error), this, [=](QProcess::ProcessError error1){Q_UNUSED(error1)});// 读取connect(m_pProcess, &QProcess::readyRead, this, [this](){if(!m_pProcess)return;QString strOutput = QString("[客户端发送输出 ] %1").arg(QString(m_pProcess->readAllStandardOutput()));ui->textBrowser->append(strOutput);});// 读取标准错误信息connect(m_pProcess, &QProcess::readyReadStandardError, this, [=](){QString strError = QString("[客户端发送错误 ] %1").arg(QString(m_pProcess->readAllStandardError()));ui->textBrowser->append(strError);});// 状态改变时触发connect(m_pProcess, &QProcess::stateChanged, this, [=](QProcess::ProcessState state){Q_UNUSED(state)});}m_pProcess->start("G:/workspace/Qt_process-invoke-client/build-client-Desktop_Qt_5_12_4_MSVC2017_64bit-Debug/debug/client.exe");
}void MainWindow::on_btn_sendValueToClient_clicked()
{if(!m_pProcess)return;// 可输中文m_pProcess->write(ui->lineEdit->text().toUtf8());
}

子进程:

#include "mainwindow.h"
#include "ui_mainwindow.h"#include <QDebug>
#include <QFile>
#include <iostream>
#include <windows.h>
#include <stdio.h>
#include <QFuture>
#include <QtConcurrent/QtConcurrent>MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);setWindowTitle("client");QFuture<void> f1 = QtConcurrent::run(this, &MainWindow::readstdin);connect(this,&MainWindow::sig_receivedCommand, this, [&](QString text){ui->textBrowser->append("[读取标准输入 ] "+ text);});connect(this, &MainWindow::sig_log, this, &MainWindow::slot_print);
}MainWindow::~MainWindow()
{delete ui;
}void MainWindow::on_btn_sendOutputToInvoker_clicked()
{QString text = ui->lineEdit->text();if(text.isEmpty()) return;
#if 0QFile fileout;if(fileout.open(stdout,QIODevice::WriteOnly)){fileout.write(text.toStdString().c_str());}else{printErr("open fail");}fileout.close();
#elsestd::cout << text.toStdString() << std::endl;
#endif
}void MainWindow::printErr(const QString &errText)
{QFile fileerr;if(fileerr.open(stderr,QIODevice::WriteOnly)){fileerr.write(errText.toStdString().c_str());}fileerr.close();
}void MainWindow::on_btn_sendErrToInvoker_clicked()
{QString text = ui->lineEdit->text();if(text.isEmpty()) return;
#if 1QFile fileerr;if(fileerr.open(stderr,QIODevice::WriteOnly)){fileerr.write(text.toStdString().c_str());}fileerr.close();
#elsestd::cerr << text.toStdString() << std::endl;
#endif
}void MainWindow::slot_print(const QString &text)
{ui->textBrowser->append("[读取标准输入 ] "+ text);
}
void MainWindow::readstdin()
{bool ok = true;char chBuf[4096];DWORD dwRead;HANDLE hStdinDup;const HANDLE hStdin = GetStdHandle(STD_INPUT_HANDLE);if (hStdin == INVALID_HANDLE_VALUE)return;DuplicateHandle(GetCurrentProcess(), hStdin,GetCurrentProcess(), &hStdinDup,0, false, DUPLICATE_SAME_ACCESS);CloseHandle(hStdin);while (ok) {ok = ReadFile(hStdinDup, chBuf, sizeof(chBuf), &dwRead, NULL);// emit sig_log(QLatin1String("ok is:")+QString::number(ok));if (ok && dwRead != 0){emit sig_receivedCommand(QString::fromUtf8(chBuf, dwRead));}}
}

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

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

相关文章

微服务设计原则——高性能:锁

文章目录 1.锁的问题2.无锁2.1 串行无锁2.2 无锁数据结构 3.减少锁竞争参考文献 1.锁的问题 高性能系统中使用锁&#xff0c;往往带来的坏处要大于好处。 并发编程中&#xff0c;锁带解决了安全问题&#xff0c;同时也带来了性能问题&#xff0c;因为锁让并发处理变成了串行操…

海外营销推广:快速创建维基百科(wiki)词条-大舍传媒

一、维基百科的永久留存问题 许多企业和个人关心维基百科是否能永久留存。实际上&#xff0c;只要企业和个人的行为没有引起维基百科管理方的反感&#xff0c;词条就可以长期保存。如果有恶意行为或被投诉&#xff0c;维基百科可能会对词条进行删除或修改。 二、创建维基百科…

TCP与UDP网络编程

网络通信协议 java.net 包中提供了两种常见的网络协议的支持: UDP&#xff1a;用户数据报协议(User Datagram Protocol)TCP&#xff1a;传输控制协议(Transmission Control Protocol) TCP协议与UDP协议 TCP协议 TCP协议进行通信的两个应用进程&#xff1a;客户端、服务端 …

好玩的调度技术-场景编辑器

好玩的调度技术-场景编辑器 文章目录 好玩的调度技术-场景编辑器前言一、演示一、代码总结好玩系列 前言 这两天写前端写上瘾了&#xff0c;顺手做了个好玩的东西&#xff0c;好玩系列也好久没更新&#xff0c;正好作为素材写一篇文章&#xff0c;我真的觉得蛮好玩的&#xff…

LinuxShell编程1———shell基础命令

文章目录 前言 一、shell基础知识 1、shell概念 2、Shell的功能 接收&#xff1a;用户命令 调用&#xff1a;相应的应用程序 解释并交给&#xff1a;内核去处理 返还&#xff1a;内核处理结果 3、Shell种类&#xff08;了解&#xff09; 3.1、MS-DOS 3.2、Windows的…

R语言进行K折交叉验证问题

在使用R语言进行模型参数评估优化时候&#xff0c;会使用K折交叉验证&#xff0c;其中会遇到各种各样问题&#xff1a; 错误: C5.0 models require a factor outcome > (1-mean(E0));(1-mean(E1)) [1] 1 [1] 1 报错说明C5.0模型需要因子变量输出&#xff0c;源代码如下&am…

无人机技术优势及发展详解

一、技术优势 无人机&#xff08;Unmanned Aerial Vehicle&#xff0c;UAV&#xff09;作为一种新兴的空中智能平台&#xff0c;凭借其独特的技术优势&#xff0c;已经在众多领域中展现出强大的应用潜力和实用价值。以下是无人机的主要技术优势&#xff1a; 1. 自主导航与远程…

【Harmony】SCU暑期实训鸿蒙开发学习日记Day2

目录 Git 参考文章 常用操作 ArkTS的网络编程 Http编程 发送请求 GET POST 处理响应 JSON数据解析 处理响应头 错误处理 Web组件 用生命周期钩子实现登录验证功能 思路 代码示例 解读 纯记录学习日记&#xff0c;杂乱&#xff0c;误点的师傅可以掉了&#x1…

How to integrate GPT-4 model hosted on Azure with the gptstudio package

题意&#xff1a;怎样将托管在Azure上的GPT-4模型与gptstudio包集成&#xff1f; 问题背景&#xff1a; I am looking to integrate the OpenAI GPT-4 model into my application. Here are the details I have: Endpoint: https://xxxxxxxxxxxxxxx.openai.azure.com/Locatio…

LG 选择 Flutter 来增强其智能电视操作系统 webOS

可以这个话题会让大多数人困惑&#xff0c;2024 年了为什么还会冒出 webOS 这种老古董&#xff1f;然后 LG 为什么选择 webOS &#xff1f;现在为什么又选择 Flutter &#xff1f; 其实早在 Google I/O 发布 Flutter 3.22 版本的时候&#xff0c;就提到了 LG 选择 Flutter 来增…

tinymce富文本支持word内容同时粘贴文字图片上传 vue2

效果图 先放文件 文件自取tinymce: tinymce富文本简单配置及word内容粘贴图片上传 封装tinymce 文件自取&#xff1a;tinymce: tinymce富文本简单配置及word内容粘贴图片上传 页面引用组件 <TinymceSimplify refTinymceSimplify v-model"knowledgeBlockItem.content…

【leetcode】 字符串相乘(大数相乘、相加)

记录一下大数相乘相加方法&#xff1a; 给定两个以字符串形式表示的非负整数 num1 和 num2&#xff0c;返回 num1 和 num2 的乘积&#xff0c;它们的乘积也表示为字符串形式。 注意&#xff1a;不能使用任何内置的 BigInteger 库或直接将输入转换为整数。 示例 1: 输入: nu…

vue3前端开发-执行npm run dev提示报错怎么解决

vue3前端开发-执行npm run dev提示报错怎么解决&#xff01;今天在本地安装初始化了一个vue3的案例demo。但是当我执行npm run dev想启动它时报错了说&#xff0c;找不到dev。让我检查package.json文件是否包含dev。如下图所示&#xff1a; 实际上&#xff0c;不必惊慌&#xf…

iOS ------ tagged Pointer 内存对齐

一&#xff0c;tagged Pointer 为了节省内存和提高执行效率&#xff0c;苹果在64bit程序中引入了Tagged Pointer计数&#xff0c;用于优化NSNumber, NSDate, NSString等小对象的存储。一个指针或地址区域&#xff0c;除了放对象地址之外&#xff0c;也可以放其他额外的信息&am…

240717.LeetCode——2974.最小数字游戏

题目描述 你有一个下标从 0 开始、长度为 偶数 的整数数组 nums &#xff0c;同时还有一个空数组 arr 。Alice 和 Bob 决定玩一个游戏&#xff0c;游戏中每一轮 Alice 和 Bob 都会各自执行一次操作。游戏规则如下&#xff1a; 每一轮&#xff0c;Alice 先从 nums 中移除一个 …

转移C盘中的conda环境(包括.condarc文件修改,environment.txt文件修改,conda报错)

conda环境一般是默认安装到C盘的&#xff0c;若建立多个虚拟环境&#xff0c;时间长了&#xff0c;容易让本不富裕的C盘更加雪上加霜&#xff0c;下面给出将conda环境从C盘转移到D盘的方法。 目录 电脑软硬件转移方法查看当前conda目录转移操作第一步&#xff1a;.condarc文件修…

Apache Flink 入门

零、概述 Apache Flink 是一个高性能的开源分布式流处理框架&#xff0c;专注于实时数据流的处理。 它设计用于处理无界和有界数据流&#xff0c;在内存级速度下提供高效的有状态计算。 Flink 凭借其独特的Checkpoint机制和Exactly-Once语义&#xff0c;确保数据处理的准确性…

只用 CSS 能玩出什么花样?

在前端开发领域&#xff0c;CSS 不仅仅是一种样式语言&#xff0c;它更像是一位多才多艺的艺术家&#xff0c;能够创造出令人惊叹的视觉效果。本文将带你探索 CSS 的无限可能&#xff0c;从基本形状到动态动画&#xff0c;从几何艺术到仿生设计&#xff0c;只用 CSS 就能玩出令…

Vscode中Github copilot插件无法使用(出现感叹号)解决方案

1、击扩展或ctrl shift x ​​​​​​​ 2、搜索查询或翻找到Github compilot 3、点击插件并再左侧点击登录github 点击Sign up for a ... 4、跳转至github登录页&#xff0c;输入令牌完成登陆后返回VScode 5、插件可以正常使用

微服务实战系列之玩转Docker(三)

前言 镜像&#xff08;Image&#xff09;作为Docker的“水源”&#xff0c;取之于它&#xff0c;用之于它。这对于立志成为运维管理的撒手锏——Docker而言&#xff0c;重要性不言而喻。 我们在虚拟机时代&#xff08;当然现在依然ing…&#xff09;&#xff0c;如何快速完成…