QT运行一个实例进程
开发项目的工程中,经常希望只运行一个相同的QT实例进程。当然也可以像类似QQ那样,同时登录多个不同的账号!为了避免这个问题,介绍以下几种用法,每一个解决方案都适用于特定的情况,所以必须尝试和适用合适的解决方案。
1. 使用QLocalSocket
方式:首先一个新的实例启动时,将尝试连接到同一个本地服务器,如果连接失败,则表示第一个实例进程,创建一个本地服务器。否则,进行退出。
void MainWidget::initLocalConnection()
{
is_running = false;
QCoreApplication::setApplicationName("localserver");
QString serverName = QCoreApplication::applicationName();
QLocalSocket socket;
socket.connectToServer(serverName);
if(socket.waitForConnected(500))
{
is_running = true;
return;
}
//连接不上服务器,就创建一个
server = new QLocalServer(this);
connect(server, SIGNAL(newConnection()), this, SLOT(newLocalConnection()));
if(server->listen(serverName))
{
//防止程序崩溃时,残留进程服务,移除之
if(server->serverError() == QAbstractSocket::AddressInUseError && QFile::exists(server->serverName()))
{
QFile::remove(server->serverName());
server->listen(serverName);
}
}
}
void MainWidget::newLocalConnection()
{
QLocalSocket *socket = server->nextPendingConnection();
if(!socket)
return;
socket->waitForReadyRead(1000);
delete socket;
}
bool MainWidget::isRunning()
{
return is_running;
}
main.cpp中:
MainWidget main_widget;
if(!main_widget.isRunning())
{
main_widget.showNormal();
main_widget.raise();
main_widget.activateWindow();
return app.exec();
}
- 创建一个QLoaclSocket,连接服务器
- 创建一个QLocalServer,并监听连接
- 启动应用程序之间,检查有多少个连接,如果至少有一个,意味着打开了一个应用程序,则不再打开另一个。
2. 使用QSharedMemory
方式:先创建一个共享内存,然后在每一个应用程序开始运行之前进行检查是否可以创建一个具有相同unique_id的共享内存,如果不能,则表示创建了实例正在运行。
QSharedMemory shared_memory;
shared_memory.setKey(unique_id);
if(shared_memory.attach())
{
return 0;
}
if(shared_memory.create(1))
{
MainWidget main_widget;
main_widget.showNormal();
main_widget.raise();
main_widget.activateWindow();
return app.exec();
}
附:方法二main.cpp中的完整代码示例
//mian.cpp
#include "mainwindow.h"
#include <QApplication>
#include <QSharedMemory>
#include <QMessageBox>
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
QSharedMemory shared_memory;
shared_memory.setKey("THISISATEST00594869");
if(shared_memory.attach())
{
QMessageBox::information(0,"Information","This program is running already.",QMessageBox::Yes);
return 0;
}
if(shared_memory.create(1))
{
MainWindow w;
w.show();
return a.exec();
}
return -1;
}
有时候我们想让程序仅运行1次,即只有一个实例生成。在Windows平台下,可以通过在main.cpp中添加CreateMutex函数来实现:
首先在main.cpp中添加调用windows API的函数:
#ifdef Q_WS_WIN
#include <windows.h>
#endif
接着在main()函数中添加:
// 测是否只运行了这一个实例
#ifdef Q_WS_WIN
HANDLE hMutex = CreateMutex(NULL, true, QString("SoftwareString").toStdWString().c_str());
if (GetLastError() == ERROR_ALREADY_EXISTS)
{
CloseHandle(hMutex);
hasRun dialog(0);
dialog.exec();
a.exit(1);
return 1;
}
#endif
其中“SoftwareString”应该是可以任意定义的,只要与其他的程序不同就行了。“hasRun”是我自己写的一个对话框,用于提醒用户这个程序已经运行了,请用户打开已经运行的实例,进行操作。
用上面这段代码可以保证程序只运行一个实例。如果用户想重复运行,则会弹出提示。我觉得最好是用户如果重复运行的话,就关闭重复的实例,然后自动打开已运行的实例。不知道这个怎么实现?
我自己开始是想用WindowsAPI里的FindWindow函数来查找已运行的实例。但是这种方法很慢。而且有时主窗口的中文标题在xp以前的系统里是乱码,FindWindow用起来不是很方便。不知道大神们有没有什么好办法?