添加链接
link之家
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接

centos下当运行一个程序,并将它最小化到系统托盘,时间久了用户可能忘了此程序还开着。而又去试图打开该程序,若是单实例会提示错误,若不是单实例则会启动新的程序(但不想运行多个程序)。怎么才能做到:直接弹出正在运行并且已经最小化在系统托盘里的程序。
解决方法:在系统托盘(QSystemTrayIcon)的程序上加上QSharedMemory判断是否是单实例(可采用QtSingleApplication更好),再加上signal信号处理(接收SIGHUP信号)。
如下:始终只有一个托盘

1、部分代码

(1)widget.h

#ifndef WIDGET_H
#define WIDGET_H
#include <QtGui>
#include<signal.h>
#include "systemtray.h"
class Widget : public QWidget
    Q_OBJECT
public:
    Widget(QWidget *parent = 0);
    ~Widget();
protected:
    void closeEvent(QCloseEvent *event);
private:
    static void showHandle(int sig);
private:
    SystemTray *systemTray;
    static Widget *instance;
#endif // WIDGET_H

(2)widget.cpp

#include "widget.h"
Widget *Widget::instance = NULL;
Widget::Widget(QWidget *parent)
    : QWidget(parent, Qt::WindowMinimizeButtonHint | Qt::WindowCloseButtonHint | Qt::WindowStaysOnTopHint)
    setWindowTitle(tr("系统托盘"));
    resize(400, 400);
    move((QApplication::desktop()->width() - width())/2,  (QApplication::desktop()->height() - height())/2);
    systemTray = new SystemTray(this);
    systemTray->setMiddleClickText(tr("系统托盘"));
    systemTray->setToolTips(tr("应用程序系统托盘"));
    systemTray->setHideText(tr("应用程序隐藏在系统托盘"));
    systemTray->show();
    instance = this;
    signal(SIGHUP, showHandle);
Widget::~Widget()
    instance = NULL;
void Widget::closeEvent(QCloseEvent *event)
    if (systemTray->isVisible()) {
        systemTray->startHideTips();
        hide();
        event->ignore();
void Widget::showHandle(int sig)
    if (instance) {
        instance->showNormal();
//    if (instance && !instance->isVisible()) {
//        instance->show();
//    }

(3)main.cpp

#include "widget.h"
#include <QApplication>
#include <QSharedMemory>
int main(int argc, char *argv[])
    QApplication a(argc, argv);
    QTextCodec *codec = QTextCodec::codecForName("utf8");
    QTextCodec::setCodecForLocale(codec);
    QTextCodec::setCodecForCStrings(codec);
    QTextCodec::setCodecForTr(codec);
    if (!QSystemTrayIcon::isSystemTrayAvailable()) {
        QMessageBox::critical(NULL, QObject::tr("系统托盘"), QObject::tr("不支持系统托盘"));
        return 1;
    QSharedMemory mem("system_tray");
    if (!mem.create(1)) {
        QString appName =  QApplication::applicationFilePath().section('/', -1);
        QProcess::startDetached(QString("killall -HUP %1").arg(appName));
        return 1;
    Widget w;
    w.show();
    return a.exec();
//    QSharedMemory shared_memory;
//    shared_memory.setKey("systemtray");
//    if(shared_memory.attach()) {
//        return 0;
//    }
//    if(shared_memory.create(1)) {
//        Widget w;
//        w.show();
//        return a.exec();
//    }

(1)上述代码已上传到CSDN:http://download.csdn.net/detail/taiyang1987912/9436271
(2)程序中QSharedMemory有比较大的隐患,每次异常中断后共享内存无法删除,测试可以修改QSharedMemory mem("system_tray")的名称(或手动删除共享内存),实际项目中最好使用QtSingleApplication。
(3)除了使用signal还可以使用QDBus,发送dbus信号到连接的槽函数,但QDBus相对来说比较麻烦点。
(4)若有问题或建议,请留言,在此感谢!

一、简介 centos下当运行一个程序,并将它最小化到系统托盘,时间久了用户可能忘了此程序还开着。而又去试图打开该程序,若是单实例会提示错误,若不是单实例则会启动新的程序(但不想运行多个程序)。怎么才能做到:直接弹出正在运行并且已经最小化在系统托盘里的程序。 解决方法:在系统托盘(QSystemTrayIcon)的程序上加上QSharedMemory判断是否是...