Qt浅谈之四十三Linux下有系统托盘再运行弹出已运行的实例
2016-02-19 10:34
477 查看
一、简介
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)若有问题或建议,请留言,在此感谢!
相关文章推荐
- Linux socket 初步
- Linux Kernel 4.0 RC5 发布!
- linux lsof详解
- linux 文件权限
- Linux 执行数学运算
- 10 篇对初学者和专家都有用的 Linux 命令教程
- Linux 与 Windows 对UNICODE 的处理方式
- Ubuntu12.04下QQ完美走起啊!走起啊!有木有啊!
- 解決Linux下Android开发真机调试设备不被识别问题
- 运维入门
- 运维提升
- Linux 自检和 SystemTap
- Ubuntu Linux使用体验
- c语言实现hashmap(转载)
- Linux 信号signal处理机制
- linux下mysql添加用户
- Scientific Linux 5.5 图形安装教程
- Linux 下无损图片压缩小工具介绍