Qt 进程和线程之一:运行一个进程和进程间通信
Qt提供了一个与平台无关的QProcess类,用以对进程的支持。本节讲述了怎样在Qt应用程序中启动一个外部程序进程,以及几种常用的进程间通信方法。如果对进程和线程的概念不是很了解,可以看我的另一篇博客:[多进程和多线程的概念。
设计应用程序时,有时不希望将一个不太相关的功能集成到程序中,或者是因为该功能与当前设计的应用程序联系不大,或者是因为该功能已经可以使用现成的程序很好地实现了,这时就可以在当前的应用程序中调用外部的程序来实现该功能,这就会使用到进程。Qt应用程序可以很容易地启动一个外部应用程序,而且Qt也提供了多种进程间通信的方法。
一、运行一个进程
Qt的QProcess类可以用来启动一个外部程序并与其进行通信。下面我们来看一下怎么在Qt代码中启动一个进程。
运行一个进程打开记事本
首先创建QtGui应用,工程名称为“myProcess”,其他选项保持默认即可。先进入mainwindow.h文件添加代码,添加私有对象定义:QProcess myProcess。然后在设计模式往界面上拖入一个Push Button部件,修改其显示文本为“启动一个进程”。在按钮上点击鼠标右键,转到其clicked()信号对应的槽,更改如下:
void MainWindow::on_pushButton_clicked()
{
myProcess.start("notepad.exe");
}
这里我们使用QProcess对象运行了Windows系统下的记事本程序(即notepad.exe程序),因为该程序在系统目录中,所以这里不需要指定其路径。大家也可以运行其他任何的程序,只需要指定其具体路径即可。我们看到,可以使用start()来启动一个程序,有时启动一个程序时需要指定启动参数,这种情况在命令行启动程序时是很常见的。
运行程序,单击界面上的按钮时就会弹出一个记事本程序。
QProcess也提供了一组函数,可以脱离事件循环来使用,它们会挂起调用的线程直到确定的信号被发射:
- waitForStarted()阻塞直到进程已经启动;
- waitForReadyRead()阻塞直到在当前读通道上有可读的数据;
- waitForBytesWritten()阻塞直到一个有效负载数据已经被写人到进程;
- waitForFinished()阻塞直到进程已经结束。
二、进程间通信方式
Qt提供了多种方法在Qt应用程序中实现进程间通信IPC(Inter-Process Communication)。 简单介绍如下 :
TCP/IP
跨平台的Qt Network模块提供的类可以让网络编程更加便携和方便。它提供了高级类(例如:QNetworkAccessManager、QFtp)通信,使用特定的应用程序级协议,和较底层的类(例如:QTcpSocket、QTcpServer、QSslSocket)用于实现协议。
共享内存
跨平台的QSharedMemory-共享内存类,提供对操作系统的共享内存的实现。它允许多个线程和进程安全访问共享内存段。此外,QSystemSemaphore可以用来控制访问由系统共享的资源,以及进程之间的通信。
D-Bus
Qt的D-Bus模块是一种可用于使用D-Bus协议实现IPC的唯一Unix库。它将Qt的信号和槽机制延伸到IPC级别,允许由一个进程发出的信号被连接到另一个进程的槽。Qt的D-Bus文档已经详细说明如何使用Qt中的D-Bus模块。
QProcess
使用先前提到的QProcess类。跨平台类QProcess可以用于启动外部程序作为子进程,并与它们进行通信。它提供了用于监测和控制该子进程状态的API。另外,QProcess为从QIODevice继承的子进程提供了输入/输出通道。
会话管理
在Linux/X11平台上,Qt提供了会话管理的支持。会话允许事件传播到进程,例如,当检测到关机时。进程和应用程序可以执行任何必要的操作,例如:保存打开的文档。
三、共享内存之进程间通信实例
下面来看一个使用共享内存的例子,它实现的功能是:先在一个对话框中将一张图片写入到共享内存段中,然后再在另一个对话框中从共享内存段读出该图片。
新建Qt Gui应用,名称为mylPC,类名为Dialog,基类选择QDialog。完成后进人设计模式,向界面中放入两个Push Button部件和一个Label部件。将一个按钮的显示文本更改为“从文件中加载图片”,将其objectName属性更改为loadFromFileButton,将另一个按钮的显示文本更改为“从共享内存显示图片”,将其objectName属性更改为loadFromSharedMemoryButton。然后进人dialog.h文件,修改如下:
#ifndef DIALOG_H
#define DIALOG_H
#include <QDialog>
#include <QSharedMemory>
namespace Ui {
class Dialog;
}
class Dialog : public QDialog
{
Q_OBJECT
public:
explicit Dialog(QWidget *parent = nullptr);
~Dialog();
public slots:
//从文件中加载图片到共享内存
void loadFromFile();
//从共享内存中加载图片
void loadFromMemory();
private slots:
//从文件中加载图片按钮
void on_loadFromFileButton_clicked();
//从共享内存加载并显示图片按钮
void on_loadFromSharedMemoryButton_clicked();
private:
//将进程与共享内存段进行分离,如果失败则进行提示
void detach();
private:
Ui::Dialog *ui;
QSharedMemory sharedMemory; //定义一个共享内存对象
};
#endif // DIALOG_H
然后进人dialog.cpp文件,修改如下:
#include "dialog.h"
#include "ui_dialog.h"
#include <QFileDialog>
#include <QBuffer>
#include <QDebug>
Dialog::Dialog(QWidget *parent) :
QDialog(parent),
ui(new Ui::Dialog)
{
ui->setupUi(this);
//在使用共享内存以前,需要设置key,系统用它作为底层共享内存段的标识
sharedMemory.setKey("QSharedMemoryExample");
}
Dialog::~Dialog()
{
delete ui;
}
//从文件中加载图片到共享内存
void Dialog::loadFromFile()
{
//判断该进程是否已经连接到共享内存段,连接成功返回true
if (sharedMemory.isAttached())
detach();
ui->label->setText(tr("选择一个图片文件!"));
//使用文件对话框获得打开图片路径
QString fileName = QFileDialog::getOpenFileName(nullptr, QString(), QString(),
tr("Images (*.png *.jpg)"));
//label标签显示图片
QImage image;
if (!image.load(fileName))
{
ui->label->setText(tr("选择的文件不是图片,请选择图片文件!"));
return;
}
ui->label->setPixmap(QPixmap::fromImage(image));
//将图片加载到共享内存
//使用QBuffer来暂存图片,这样便可以获得图片的大小,还可以获得图片数据的指针
QBuffer buffer;
buffer.open(QBuffer::ReadWrite);
QDataStream out(&buffer);
out << image;
int size = static_cast<int>(buffer.size());
//使用create()函数创建指定大小的共享内存段,该函数还会自动将共享内存段连接到本进程上。
if (!sharedMemory.create(size))
{
ui->label->setText(tr("无法创建共享内存段!"));
return;
}
//在进行共享内存段的操作前,需要先进行加锁
sharedMemory.lock();
char *to = static_cast<char*>(sharedMemory.data());
const char *from = buffer.data().data();
//使用memcpy()函数将图片数据复制到共享内存
memcpy(to, from, static_cast<size_t>(qMin(sharedMemory.size(), size)));
//等操作完成后,再进行解锁。
sharedMemory.unlock();
}
//从共享内存中加载图片
void Dialog::loadFromMemory()
{
//使用attache()函数将进程连接到共享内存段
if (!sharedMemory.attach())
{
ui->label->setText(tr("无法连接到共享内存段,\n"
"请先加载一张图片!"));
return;
}
QBuffer buffer;
QDataStream in(&buffer);
QImage image;
//使用QBuffer来读取共享内存段中的数据
sharedMemory.lock();
buffer.setData((char*)sharedMemory.constData(), sharedMemory.size());
buffer.open(QBuffer::ReadOnly);
in >> image;
sharedMemory.unlock();
//将进程与共享内存段进行分离,如果失败则进行提示
sharedMemory.detach();
ui->label->setPixmap(QPixmap::fromImage(image));
}
//将进程与共享内存段进行分离,如果失败则进行提示
void Dialog::detach()
{
if (!sharedMemory.detach())
ui->label->setText(tr("无法从共享内存中分离!"));
}
//从文件中加载图片按钮
void Dialog::on_loadFromFileButton_clicked()
{
loadFromFile();
}
//从共享内存加载并显示图片按钮
void Dialog::on_loadFromSharedMemoryButton_clicked()
{
loadFromMemory();
}
在一个运行的实例上单击“从文件中加载图片”按钮,然后选择一张图片。在第二个运行的实例上单击“从共享内存显示图片”按 钮,这时便会显示第一个实例中加载的图片,效果如下所示:
四、程序分析
(1)loadFromFile()
- 这里先使用isAttached()函数判断该进程是否已经连接到共享内存段,如果是,那么就调用detach()先将该进程与共享内存段进行分离。
- 然后使用QFileDialog类来打开一个图片文件,并将其显示到标签上。为了将图片加载到共享内存,这里使用了QBuffer来暂存图片,这样便可以获得图片的大小,还可以获得图片数据的指针。
- 后面使用了create()函数来创建指定大小的共享内存段,其大小的单位是字节,该函数还会自动将共享内存段连接到本进程上。
- 在操作共享内存段时要使用lock()进行加锁,然后才可以使用memcpy()函数将buffer对应的数据段复制到共享内存段,操作完成后要使用unlock()进行解锁。这样在同一时间,就只能有一个进程允许操作共享内存段了。
(2)loadFromMemory()
- 这里先使用attache()函数将进程连接到共享内存段。
- 在操作共享内存段时要使用lock()进行加锁,然后才可以使用QBuffer来读取共享内存段中的数据,操作完成后要使用unlock()进行解锁。
- 因为现在已经不需要使用共享内存了,所以调用detach()函数将进程与共享内存段进行分离。最后将图片显示到标签中。
参考:
Qt 进程和线程之一:运行一个进程和进程间通信的更多相关文章
- C# 最基本的涉及模式(单例模式) C#种死锁:事务(进程 ID 112)与另一个进程被死锁在 锁 | 通信缓冲区 资源上,并且已被选作死锁牺牲品。请重新运行该事务,解决方案: C#关闭应用程序时如何关闭子线程 C#中 ThreadStart和ParameterizedThreadStart区别
C# 最基本的涉及模式(单例模式) //密封,保证不能继承 public sealed class Xiaohouye { //私有的构造函数,保证外部不能实例化 private ...
- Kettle 解决数据锁的问题(事务(进程 ID 51)与另一个进程被死锁在 锁 资源上)
1.Kettle做了一个作业, 执行的时候问题发生在步骤2和步骤3之间,也就是步骤2还未完全执行完的时候,步骤3就要更新步骤2插入的数据,造成死锁.(我的理解是既然都分开作业了,那么每个作业都是一个单 ...
- 并发错误:事务(进程 ID )与另一个进程已被死锁在 lock 资源上,且该事务已被选作死锁牺牲品
这个是并发情况下导致的数据库事务错误,先介绍下背景. 背景 springboot+springmvc+sqlserver+mybatis 一个controller里有五六个接口,这些接口都用到了spr ...
- 事务(进程 ID 64)与另一个进程被死锁在 锁 资源上,并且已被选作死锁牺牲品。
访问频率比较高的app接口,在后台写的异常日志会偶尔出现以下错误. 事务(进程 ID 64)与另一个进程被死锁在 锁 资源上,并且已被选作死锁牺牲品.请重新运行该事务 实所有的死锁最深层的原因就是一个 ...
- SQL Server死锁问题:事务(进程 ID x)与另一个进程被死锁在 锁 | 通信缓冲区资源上并且已被选作死锁牺牲品。请重新运行该事务。
### The error occurred while setting parameters### SQL: update ERP_SCjh_zzc_pl set IF_TONGBU=1 where ...
- LR_问题_虚拟用户以进程和线程模式运行的区别
进程方式和线程方式的优缺点: 如果选择按照进程方式运行, 每个用户都将启动一个mmdrv进程,多个mmdrv进程会占用大量内存及其他系统资源,这就限制了可以在任一负载生成器上运行的并发用户数的数量,因 ...
- LR-虚拟用户以进程和线程模式运行的区别
进程方式和线程方式的优缺点: 如果选择按照进程方式运行, 每个用户都将启动一个mmdrv进程,多个mmdrv进程会占用大量内存及其他系统资源,这就限制了可以在任一负载生成器上运行的并发用户数的数量,因 ...
- Java并发编程:进程和线程之由来__进程让操作系统的并发性成为可能,而线程让进程的内部并发成为可能
转载自海子:http://www.cnblogs.com/dolphin0520/p/3910667.html Java多线程基础:进程和线程之由来 在前面,已经介绍了Java的基础知识,现在我们来讨 ...
- Linux下的进程与线程(一)—— 进程概览
进程是操作系统分配资源的基本单位.线程是操作系统进行运行和调度的基本单位. 进程之间可以切换,以便轮流占用CPU,实现并发.一般进程运行在用户模式下,只能执行指令集中的部分指令. 当进程进行上下文切换 ...
随机推荐
- git项目.gitignore文件不生效解决办法
配置好.gitignore文件如下: HELP.md /target/ !.mvn/wrapper/maven-wrapper.jar ### STS ### .apt_generated .clas ...
- windows64位安装mysql-5.7.12,图文
linux下安装mysql教程一大片,我就不说了,再此说下windows 下如何安装这个5.7版本,并且有些坑已踩! 一:进入mysql下载地址:http://www.mysql.com/downlo ...
- 转载的C#学习笔记
转载地址:http://www.cnblogs.com/renyanlei/p/4075065.html 最近在一个培训机构里面教授Net知识.每天都会带领学生学习c#知识.我希望把每天学习的笔记记录 ...
- UIButton设置为圆形按钮并增加边框
设置按钮的长和宽尺寸一致(即为正方形),然后将圆角半径设为边长的一半,即形成一个圆形 UIButton *btn = [UIButton buttonWithType:UIButtonTypeSyst ...
- CentOS环境 升级Python2.6.6至2.7.5
1.查看当前Python版本 # python -V Python 2.6.6 # python -V Python 2.6.6 2.下载Python2.7.5源码 # wget http://p ...
- BZOJ_1044_[HAOI2008]木棍分割_二分答案+DP+单调队列
BZOJ_1044_[HAOI2008]木棍分割_二分答案+DP Description 有n根木棍, 第i根木棍的长度为Li,n根木棍依次连结了一起, 总共有n-1个连接处. 现在允许你最多砍断m个 ...
- No result defined for action cn.crm.action.LinkManAction and result input
这是struts2的一个拦截器报的错误,当你的form中的数据有问题,比如说<input type="text" name="receiverLoginID&quo ...
- HDU2159(完全背包)
FATE Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u Description ...
- HDU2203(KMP入门题)
亲和串 Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submiss ...
- Gulp简单应用
1.创建一个工程,在webstorm控制台 cnpm install --save-dev gulp cnpm install --save-dev gulp-concat ...