http://blog.51cto.com/9291927/1879762
Qt 是一个跨平台的应用框架,其进程间通信机制当然可以使用所在平台的进程间通信机制,如在Windows平台上的Message机制、共享内存、文件映射、管道、Socket等。其中,Qt对一些许多平台共有的IPC机制进行了封装。
一、TCP/IP
其实就是通过网络模块实现的IPC。不过Qt对其进行了封装,并提供了两个层次的API,包括应用程序级的QNetworkAccessManager, QFtp等和底层的QTcpSocket, QTcpServer, QSslSocket等。
二、QShared Memory(共享内存)
Qt提供的基于共享内存的IPC有QSharedMemory类和QSystemSemaphore类,QSharedMemory可以访问共享内存区域,以及多线程和进程的共享内存区域,而QSystemSemaphore类用于访问系统共享资源,以实现独立进程间的通信。
1、QSharedMemory
QSharedMemory读写内存时,可以使用lock()实现同步。如果同步完成,必须使用unlock()为共享内存区域解锁。
QSharedMemory可以使用attach()访问共享内存,通过指定参数来设置共享内存的访问模式。如果使用的是QSharedMemory::ReadOnly模式,则只能通过只读模式访问共享内存,如果使用QSharedMemory::ReadWrite模式则可以通过读写模式访问共享内存。
QSharedMemory拥有进程并提供可以返回共享内存区域指针的成员函数。在共享内存区域,成员函数constData()可以通过void 类型返回进程正在使用的内存区域指针。创建共享时,QSharedMemory可以以字节为单位分配共享内存区域,还可以通过第二个参数设置函数 attach()提供的模式。
QSharedMemory可以设置特定共享内存的固定键。函数setNativeKey()可以设置共享内存对象的键,setNativeKey()函数使用从属平台的共享内存的键进行相关设置。使用函数setKey()可以设置独立于平台的键。函数setKey()创建与平台本地键(Native Key)映射的键。
初始化QSharedMemory时,必须指定一个唯一的标识Key,进程的Key必须保持一致。可以使用setKey来设置。
2、QSystemSemaphore
QSystemSemaphore可以提供普通系统的信号量。信号量使用互斥体,而互斥体只可以使用1次锁定(Block)。因此,QSemaphore类不能对有效资源使用多线程,而QSystemSemaphore类可以再多进程或多线程中实现。
QSystemSemaphore与QSemaphore类不同,可以访问多进程。这表示QSystemSemaphore是一个重量级的类。因 此,使用单一线程或进程时,建议使用QSemaphore。获得资源前,成员函数acquire()始终阻塞。函数release()用于释放资源,且该 函数可以设置参数。该函数的参数>1时,释放资源。
3、QSharedMemory编程流程
共享内存中数据提供方:
A、定义QSharedMemory shareMemory,并设置标志名shareMemory.setKey();
B、将共享内存与主进程分离 shareMemory.detach();
C、创建共享内存 shareMemory.create();
D、将共享内存上锁shareMemory.lock();
E、将进程中要共享的数据拷贝到共享内存中;
F、将共享内存解锁shareMemory.unlock();
共享内存中数据使用方:
A、定义QSharedMemory shareMemory,并设置与共享内存提供方一致的标志名shareMemory.setKey()。
B、将共享内存上锁shareMemory.lock();
C、将共享内存与主进程绑定shareMemory.attach(),使主进程可以访问共享内存的数据;
D、从共享内存中取数据;
E、使用完后将共享内存解锁shareMemory.unlock(),并将共享内存与进程分离shareMemory.detach();
4、QSharedMemory使用实例
共享内存数据提供方代码:
Widget.h文件:
#ifndef WIDGET_H
#define WIDGET_H
#include <QtGui/QWidget>
#include <QSharedMemory>
class Widget : public QWidget
{
Q_OBJECT
public:
Widget(QWidget *parent = 0);
~Widget();
private:
QSharedMemory sharememory;
private slots:
void WriteShareMemory();
};
#endif // WIDGET_H
Widget.cpp文件:
#include "Widget.h"
#include <QBuffer>
#include <QDebug>
#include <QDataStream>
Widget::Widget(QWidget *parent)
: QWidget(parent)
{
sharememory.setKey("share");
this->WriteShareMemory();
}
Widget::~Widget()
{
}
void Widget::WriteShareMemory()
{
if(sharememory.isAttached())
{
sharememory.detach();
}
QBuffer buffer;
QDataStream out(&buffer);
buffer.open(QBuffer::ReadWrite);
buffer.write("hello QT!");
int size = buffer.size();
if(!sharememory.create(size))
{
qDebug() << sharememory.errorString();
return ;
}
sharememory.lock();
char *dest = reinterpret_cast<char *>(sharememory.data());
const char *source = reinterpret_cast<const char *>(buffer.data().data());
memcpy(dest, source, qMin(size, sharememory.size()));
sharememory.unlock();
}
Main.cpp文件:
#include <QtGui/QApplication>
#include "Widget.h"
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
Widget w;
w.show();
return a.exec();
}
共享内存数据使用方代码:
Widget.h文件:
#ifndef WIDGET_H
#define WIDGET_H
#include <QtGui/QWidget>
#include <QSharedMemory>
class Widget : public QWidget
{
Q_OBJECT
public:
Widget(QWidget *parent = 0);
~Widget();
private:
QSharedMemory sharememory;
private slots:
void ReadShareMemory();
};
#endif // WIDGET_H
Widget.cpp文件:
#include "Widget.h"
#include <QBuffer>
#include <QDebug>
#include <QDataStream>
Widget::Widget(QWidget *parent)
: QWidget(parent)
{
sharememory.setKey("share");
this->ReadShareMemory();
}
Widget::~Widget()
{
}
void Widget::ReadShareMemory()
{
if(!sharememory.attach())
{
qDebug() << "cann't attach sahrememory!";
}
QBuffer buffer;
sharememory.lock();
buffer.setData((char*)sharememory.constData(),sharememory.size());
buffer.open(QBuffer::ReadWrite);
buffer.readAll();
sharememory.unlock();
sharememory.detach();
qDebug() << buffer.data().data();
}
Main.cpp文件:
#include <QtGui/QApplication>
#include "Widget.h"
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
Widget w;
w.show();
return a.exec();
}