简述
上一节中,我们分享下如何利用Windows消息机制来进行不同进程间的通信。但是有很多局限性,比如:不能跨平台,而且必须两个进程同时存在才可以,要么进程A发了消息谁接收呢?
下面我们来分享另外一种跨平台的进行间通信的方式-Shared Memory(共享内存)。
注意事项
初始化QSharedMemory时,必须指定一个唯一的标识Key,进程的Key必须保持一致。可以使用setKey来设置。
加载进内存
说明
进程A-写
分为下面几步:
- 检测该进程是否连接到共享内存段,如果连接,则将该进程与共享内存段分离。
- 从系统足够大的内存中得到一个新的共享内存段。
- 锁定该共享内存段,以阻止第二个对话框进程访问,将缓冲区中的图片复制进共享内存段。
- 将共享内存段解锁,然后第二个对话框进程就可以访问了。
实现
<code class="language-Qt hljs cs has-numbering"><span class="hljs-keyword">void Dialog::loadFromFile() { <span class="hljs-keyword">if (sharedMemory.isAttached()) { <span class="hljs-comment">// 将该进程与共享内存段分离 <span class="hljs-keyword">if (!sharedMemory.detach()) qDebug() << <span class="hljs-string">"Unable to detach from shared memory."; } QString fileName = QFileDialog::getOpenFileName(<span class="hljs-number">0, QString(), QString(), tr(<span class="hljs-string">"Images (*.png *.xpm *.jpg)")); QImage image; <span class="hljs-keyword">if (!image.load(fileName)) { qDebug() << <span class="hljs-string">"Selected file is not an image, please select another."; <span class="hljs-keyword">return; } <span class="hljs-comment">// 将数据加载到共享内存中 QBuffer buffer; buffer.open(QBuffer::ReadWrite); QDataStream <span class="hljs-keyword">out(&buffer); <span class="hljs-keyword">out << image; <span class="hljs-keyword">int size = buffer.size(); <span class="hljs-comment">// 创建共享内存段 <span class="hljs-keyword">if (!sharedMemory.create(size)) { qDebug() << sharedMemory.errorString() << <span class="hljs-string">"\n Unable to create shared memory segment."; <span class="hljs-keyword">return; } sharedMemory.<span class="hljs-keyword">lock(); <span class="hljs-keyword">char *to = (<span class="hljs-keyword">char*)sharedMemory.data(); <span class="hljs-keyword">const <span class="hljs-keyword">char *<span class="hljs-keyword">from = buffer.data().data(); memcpy(to, <span class="hljs-keyword">from, qMin(sharedMemory.size(), size)); sharedMemory.unlock(); }</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code>
从内存中读取
说明
进程B-读
分为下面几步:
- 将该进程与进程A创建的共享内存段绑定
- 锁定共享内存段,复制数据到缓冲区,然后写入到QImage中。
- 将共享内存段解锁,然后将该进程与共享内存段分离。
实现
<code class="language-Qt hljs lasso has-numbering"><span class="hljs-literal">void MainWindow<span class="hljs-tag">::loadFromMemory() { <span class="hljs-comment">// 将共享内存与该进程绑定 <span class="hljs-keyword">if (<span class="hljs-subst">!sharedMemory<span class="hljs-built_in">.attach()) { qDebug() <span class="hljs-subst"><< <span class="hljs-string">"Unable to attach to shared memory segment."; <span class="hljs-keyword">return; } <span class="hljs-comment">// 从共享内存中读取数据 QBuffer buffer; QDataStream <span class="hljs-keyword">in(<span class="hljs-subst">&buffer); QImage image; sharedMemory<span class="hljs-built_in">.lock(); buffer<span class="hljs-built_in">.setData((char<span class="hljs-subst">*)sharedMemory<span class="hljs-built_in">.constData(), sharedMemory<span class="hljs-built_in">.size()); buffer<span class="hljs-built_in">.open(QBuffer<span class="hljs-tag">::ReadOnly); <span class="hljs-keyword">in <span class="hljs-subst">>> image; sharedMemory<span class="hljs-built_in">.unlock(); sharedMemory<span class="hljs-built_in">.detach(); m_pLabel<span class="hljs-subst">->setPixmap(QPixmap<span class="hljs-tag">::fromImage(image)); }</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code>
原文作者:一去丶二三里
作者博客:去作者博客空间