QSingleApplication
? QMutex
? QSharedMemory
? Windows、OSX、Linux (Ubuntu) でスムーズに動作するものを探しています。Qt 4.7.1 の使用
質問する
30816 次
7 に答える
83
あなたが望むことをする簡単な解決策。ネットワークに依存QtSingleApplication
せず ( として)、オーバーヘッドもありません。
使用法:
int main()
{
RunGuard guard( "some_random_key" );
if ( !guard.tryToRun() )
return 0;
QAppplication a(/*...*/);
// ...
}
RunGuard.h
#ifndef RUNGUARD_H
#define RUNGUARD_H
#include <QObject>
#include <QSharedMemory>
#include <QSystemSemaphore>
class RunGuard
{
public:
RunGuard( const QString& key );
~RunGuard();
bool isAnotherRunning();
bool tryToRun();
void release();
private:
const QString key;
const QString memLockKey;
const QString sharedmemKey;
QSharedMemory sharedMem;
QSystemSemaphore memLock;
Q_DISABLE_COPY( RunGuard )
};
#endif // RUNGUARD_H
RunGuard.cpp
#include "RunGuard.h"
#include <QCryptographicHash>
namespace
{
QString generateKeyHash( const QString& key, const QString& salt )
{
QByteArray data;
data.append( key.toUtf8() );
data.append( salt.toUtf8() );
data = QCryptographicHash::hash( data, QCryptographicHash::Sha1 ).toHex();
return data;
}
}
RunGuard::RunGuard( const QString& key )
: key( key )
, memLockKey( generateKeyHash( key, "_memLockKey" ) )
, sharedmemKey( generateKeyHash( key, "_sharedmemKey" ) )
, sharedMem( sharedmemKey )
, memLock( memLockKey, 1 )
{
memLock.acquire();
{
QSharedMemory fix( sharedmemKey ); // Fix for *nix: http://habrahabr.ru/post/173281/
fix.attach();
}
memLock.release();
}
RunGuard::~RunGuard()
{
release();
}
bool RunGuard::isAnotherRunning()
{
if ( sharedMem.isAttached() )
return false;
memLock.acquire();
const bool isRunning = sharedMem.attach();
if ( isRunning )
sharedMem.detach();
memLock.release();
return isRunning;
}
bool RunGuard::tryToRun()
{
if ( isAnotherRunning() ) // Extra check
return false;
memLock.acquire();
const bool result = sharedMem.create( sizeof( quint64 ) );
memLock.release();
if ( !result )
{
release();
return false;
}
return true;
}
void RunGuard::release()
{
memLock.acquire();
if ( sharedMem.isAttached() )
sharedMem.detach();
memLock.release();
}
于 2015-01-27T13:53:13.563 に答える
8
QtSingleApplication
比較的時代遅れで、もうメンテナンスされていないので、 SingleApplication と呼ばれる代替品を書きました。
に基づいてQSharedMemory
おり、これを使用してQLocalServer
、新しいインスタンスが生成されていることを親プロセスに通知します。すべてのプラットフォームで動作し、互換性があり、Qt 5 をサポートします。
完全なコードとドキュメントは、こちらから入手できます。
基本的な例:
int main(int argc, char *argv[])
{
SingleApplication app( argc, argv );
return app.exec();
}
高度な例
とりわけ、新しく生成されたインスタンスとプライマリ インスタンス間のメッセージの送信をサポートします。次に例を示します。
int main(int argc, char *argv[])
{
SingleApplication app( argc, argv, true );
if( app.isSecondary() ) {
app.sendMessage( app.arguments().join(' ')).toUtf8() );
app.exit( 0 );
}
return app.exec();
}
于 2016-01-14T20:56:34.110 に答える
2
特定のキーで使用QSharedMemory
して、そのキーで共有メモリを作成できたかどうかを確認できます。作成できない場合、インスタンスはすでに実行されています。
QSharedMemory sharedMemory;
sharedMemory.setKey("MyApplicationKey");
if (!sharedMemory.create(1))
{
QMessageBox::warning(this, tr("Warning!"), tr("An instance of this application is running!") );
exit(0); // Exit already a process running
}
于 2014-11-22T06:34:52.093 に答える
1
Windows の場合:
HANDLE g_app_mutex = NULL;
bool check_one_app_instance()
{
g_app_mutex = ::CreateMutex(NULL, FALSE, L"8BD290769B404A7816985M9E505CF9AD64"); // this any different key as string
if(GetLastError() == ERROR_ALREADY_EXISTS)
{
CloseHandle(g_app_mutex);
return false;
}
return true;
}
于 2016-02-28T12:51:55.890 に答える