• 10843阅读
  • 15回复

Qt编写守护程序保证程序一直运行(开源) [复制链接]

上一主题 下一主题
离线liudianwu
 

图酷模式  只看楼主 倒序阅读 楼主  发表于: 2019-03-02
没有任何人敢保证自己写的程序没有任何BUG,尤其是在商业项目中,程序量越大,复杂度越高,出错的概率越大,尤其是现场环境千差万别,和当初本地电脑测试环境很可能不一样,有很多特殊情况没有考虑到,如果需要保证程序7*24小时运行,则需要想一些办法能够让程序死了能够活过来,在嵌入式linux上,大部分会采用看门狗的形式来处理,程序打开看门狗驱动后,定时喂狗,一旦超过规定的时间,则硬件软复位等。这种方式相对来说比较可靠,如果需要在普通PC机上运行怎办呢?本篇文章提供一个软件实现守护进程的办法,原理就是udp通信,单独写个守护进程程序,专门负责检测主程序是否存在,不存在则启动。主程序只需要启动live类监听端口,收到hello就回复ok就行。
为了使得兼容任意程序,特意提炼出来共性,增加了多种设置。
1:可设置检测的程序名称。
2:可设置udp通信端口。
3:可设置超时次数。
4:自动记录已重启次数。
5:自动记录最后一次重启时间。
6:是否需要重新刷新桌面。
7:可重置当前重启次数和最后重启时间。
8:自动隐藏的托盘运行或者后台运行。
9:提供界面设置程序名称已经开启和暂停服务。
代码下载 live.zip (64 K) 下载次数:647
window.open('http://www.qtcn.org/bbs/attachment/Mon_1903/44_110085_fffb63ce70b4bcb.png?11');" style="max-width:700px;max-height:700px;" onload="if(is_ie6&&this.offsetWidth>700)this.width=700;" >

守护进程核心代码:
  1. #pragma execution_character_set("utf-8")
  2. #include "frmmain.h"
  3. #include "ui_frmmain.h"
  4. #include "qtimer.h"
  5. #include "qudpsocket.h"
  6. #include "qsharedmemory.h"
  7. #include "qprocess.h"
  8. #include "qdatetime.h"
  9. #include "qapplication.h"
  10. #include "qdesktopservices.h"
  11. #include "qmessagebox.h"
  12. #if (QT_VERSION > QT_VERSION_CHECK(5,0,0))
  13. #include "qstandardpaths.h"
  14. #endif
  15. #include "app.h"
  16. frmMain::frmMain(QWidget *parent) : QWidget(parent), ui(new Ui::frmMain)
  17. {
  18.     ui->setupUi(this);
  19.     this->initForm();
  20. }
  21. frmMain::~frmMain()
  22. {
  23.     delete ui;
  24. }
  25. void frmMain::changeEvent(QEvent *event)
  26. {
  27.     //隐藏当前界面,最小化到托盘
  28.     if(event->type() == QEvent::WindowStateChange) {
  29.         if(windowState() & Qt::WindowMinimized) {
  30.             hide();
  31.         }
  32.     }
  33.     QWidget::changeEvent(event);
  34. }
  35. void frmMain::initForm()
  36. {
  37.     count = 0;
  38.     ok = false;
  39.     //每秒钟定时询问心跳
  40.     timerHeart = new QTimer(this);
  41.     timerHeart->setInterval(2000);
  42.     connect(timerHeart, SIGNAL(timeout()), this, SLOT(sendHearData()));
  43.     //从6050端口开始,如果绑定失败则将端口加1,直到绑定成功
  44.     udp = new QUdpSocket(this);
  45.     int port = 6050;
  46.     while(!udp->bind(port)) {
  47.         port++;
  48.     }
  49.     connect(udp, SIGNAL(readyRead()), this, SLOT(readData()));
  50.     if (App::TargetAppName.isEmpty()) {
  51.         ui->btnStart->setText("启动");
  52.         ui->btnStart->setEnabled(false);
  53.         timerHeart->stop();
  54.     } else {
  55.         ui->btnStart->setText("暂停");
  56.         ui->btnStart->setEnabled(true);
  57.         timerHeart->start();
  58.     }
  59.     ui->txtAppName->setText(App::TargetAppName);
  60.     ui->txtAppName->setFocus();
  61. }
  62. void frmMain::sendHearData()
  63. {
  64.     udp->writeDatagram("hello", QHostAddress::LocalHost, App::TargetAppPort);
  65.     //判断当前是否没有回复
  66.     if (!ok) {
  67.         count++;
  68.     } else {
  69.         count = 0;
  70.         ok = false;
  71.     }
  72.     //如果超过规定次数没有收到心跳回复,则超时重启
  73.     if (count >= App::TimeoutCount) {
  74.         timerHeart->stop();
  75.         QSharedMemory mem(App::TargetAppName);
  76.         if (!mem.create(1)) {
  77.             killApp();
  78.         }
  79.         QTimer::singleShot(1000 , this, SLOT(killOther()));
  80.         QTimer::singleShot(3000 , this, SLOT(startApp()));
  81.         QTimer::singleShot(4000 , this, SLOT(startExplorer()));
  82.     }
  83. }
  84. void frmMain::killApp()
  85. {
  86.     QProcess *p = new QProcess;
  87.     p->start(QString("taskkill /im %1.exe /f").arg(App::TargetAppName));
  88. }
  89. void frmMain::killOther()
  90. {
  91.     QProcess *p = new QProcess;
  92.     p->start(QString("taskkill /im %1.exe /f").arg("WerFault"));
  93.     //重建缓存,彻底清除托盘图标
  94.     if (App::ReStartExplorer) {
  95.         QProcess *p1 = new QProcess;
  96.         p1->start("taskkill /f /im explorer.exe");
  97.     }
  98. }
  99. void frmMain::startApp()
  100. {
  101.     if (ui->btnStart->text() == "开始" || ui->btnStart->text() == "启动") {
  102.         count = 0;
  103.         return;
  104.     }
  105.     QProcess *p = new QProcess;
  106.     p->start(QString("\"%1/%2.exe\"").arg(qApp->applicationDirPath()).arg(App::TargetAppName));
  107.     count = 0;
  108.     ok = true;
  109.     timerHeart->start();
  110.     App::ReStartCount++;
  111.     App::ReStartLastTime = QDateTime::currentDateTime().toString("yyyy-MM-dd HH:mm:ss");
  112.     App::writeConfig();
  113.     ui->labCount->setText(QString("已重启 %1 次").arg(App::ReStartCount));
  114.     ui->labInfo->setText(QString("最后一次重启在 %1").arg(App::ReStartLastTime));
  115. }
  116. void frmMain::startExplorer()
  117. {
  118.     //取得操作系统目录路径,指定操作系统目录下的explorer程序,采用绝对路径,否则在64位操作系统下无效
  119. #if (QT_VERSION > QT_VERSION_CHECK(5,0,0))
  120.     QString str = QStandardPaths::writableLocation(QStandardPaths::ApplicationsLocation);
  121. #else
  122.     QString str = QDesktopServices::storageLocation(QDesktopServices::ApplicationsLocation);
  123. #endif
  124.     if (App::ReStartExplorer) {
  125.         str = QString("%1\\Windows\\explorer.exe").arg(str.mid(0, 2));
  126.         QProcess *p = new QProcess(this);
  127.         p->start(str);
  128.     }
  129. }
  130. void frmMain::readData()
  131. {
  132.     QByteArray tempData;
  133.     do {
  134.         tempData.resize(udp->pendingDatagramSize());
  135.         udp->readDatagram(tempData.data(), tempData.size());
  136.         QString data = QLatin1String(tempData);
  137.         if (data.right(2) == "OK") {
  138.             count = 0;
  139.             ok = true;
  140.         }
  141.     } while (udp->hasPendingDatagrams());
  142. }
  143. void frmMain::on_btnOk_clicked()
  144. {
  145.     App::TargetAppName = ui->txtAppName->text();
  146.     if (App::TargetAppName == "") {
  147.         QMessageBox::critical(this, "提示", "应用程序名称不能为空!");
  148.         ui->txtAppName->setFocus();
  149.         return;
  150.     }
  151.     App::writeConfig();
  152.     ui->btnStart->setEnabled(true);
  153. }
  154. void frmMain::on_btnStart_clicked()
  155. {
  156.     count = 0;
  157.     if (ui->btnStart->text() == "暂停") {
  158.         timerHeart->stop();
  159.         ui->btnStart->setText("开始");
  160.     } else {
  161.         timerHeart->start();
  162.         ui->btnStart->setText("暂停");
  163.     }
  164. }
  165. void frmMain::on_btnReset_clicked()
  166. {
  167.     App::ReStartCount = 0;
  168.     App::ReStartLastTime = "2019-01-01 12:00:00";
  169.     App::writeConfig();
  170.     ui->txtAppName->setText(App::TargetAppName);
  171.     ui->labCount->setText(QString("已重启 %1 次").arg(App::ReStartCount));
  172.     ui->labInfo->setText(QString("最后一次重启在 %1").arg(App::ReStartLastTime));
  173.     QMessageBox::information(this, "提示", "重置配置文件成功!");
  174. }
主程序使用核心代码:
  1. #include "applive.h"
  2. #include "qmutex.h"
  3. #include "qudpsocket.h"
  4. #include "qstringlist.h"
  5. #include "qapplication.h"
  6. #include "qdatetime.h"
  7. #include "qdebug.h"
  8. #define TIMEMS qPrintable(QTime::currentTime().toString("HH:mm:ss zzz"))
  9. QScopedPointer<AppLive> AppLive::self;
  10. AppLive *AppLive::Instance()
  11. {
  12.     if (self.isNull()) {
  13.         QMutex mutex;
  14.         QMutexLocker locker(&mutex);
  15.         if (self.isNull()) {
  16.             self.reset(new AppLive);
  17.         }
  18.     }
  19.     return self.data();
  20. }
  21. AppLive::AppLive(QObject *parent) : QObject(parent)
  22. {
  23.     udpServer  = new QUdpSocket(this);
  24.     QString name = qApp->applicationFilePath();
  25.     QStringList list = name.split("/");
  26.     appName = list.at(list.count() - 1).split(".").at(0);
  27. }
  28. void AppLive::readData()
  29. {
  30.     QByteArray tempData;
  31.     do {
  32.         tempData.resize(udpServer->pendingDatagramSize());
  33.         QHostAddress sender;
  34.         quint16 senderPort;
  35.         udpServer->readDatagram(tempData.data(), tempData.size(), &sender, &senderPort);
  36.         QString data = QLatin1String(tempData);
  37.         if (data == "hello") {
  38.             udpServer->writeDatagram(QString("%1OK").arg(appName).toLatin1(), sender, senderPort);
  39.         }
  40.     } while (udpServer->hasPendingDatagrams());
  41. }
  42. bool AppLive::start(int port)
  43. {
  44.     bool ok = udpServer->bind(port);
  45.     if (ok) {
  46.         connect(udpServer, SIGNAL(readyRead()), this, SLOT(readData()));
  47.         qDebug() << TIMEMS << "Start AppLive Ok";
  48.     }
  49.     return ok;
  50. }
  51. void AppLive::stop()
  52. {
  53.     udpServer->abort();
  54.     disconnect(udpServer, SIGNAL(readyRead()), this, SLOT(readData()));
  55. }



欢迎关注微信公众号:Qt实战 (各种开源作品、经验整理、项目实战技巧,专注Qt/C++软件开发,视频监控、物联网、工业控制、嵌入式软件、国产化系统应用软件开发)QQ:517216493  WX:feiyangqingyun  QQ群:751439350
离线九重水

只看该作者 1楼 发表于: 2019-03-02
没电,电脑挂了,连自己都守护不了,看你怎么守护别人
离线liudianwu

只看该作者 2楼 发表于: 2019-03-02
回 九重水 的帖子
九重水:没电,电脑挂了,连自己都守护不了,看你怎么守护别人[表情] [表情] [表情] [表情]  (2019-03-02 16:19) 

来你过来,我保证不打屎你!
欢迎关注微信公众号:Qt实战 (各种开源作品、经验整理、项目实战技巧,专注Qt/C++软件开发,视频监控、物联网、工业控制、嵌入式软件、国产化系统应用软件开发)QQ:517216493  WX:feiyangqingyun  QQ群:751439350
离线leytou

只看该作者 3楼 发表于: 2019-03-02
感谢大佬分享
离线xzp21st

只看该作者 4楼 发表于: 2019-03-02
回 九重水 的帖子
九重水:没电,电脑挂了,连自己都守护不了,看你怎么守护别人[表情] [表情] [表情] [表情]  (2019-03-02 16:19) 

你这种情况得考虑上ups了
离线wcp520

只看该作者 5楼 发表于: 2019-03-03
感谢大佬分享
离线九重水

只看该作者 6楼 发表于: 2019-03-04
回 xzp21st 的帖子
xzp21st:你这种情况得考虑上ups了 (2019-03-02 18:06) 

正解。
离线liuchangyin

只看该作者 7楼 发表于: 2019-03-04
这个是保证软件运行的最后一道保险,我也曾这样做过
离线liudianwu

只看该作者 8楼 发表于: 2019-03-04
回 liuchangyin 的帖子
liuchangyin:这个是保证软件运行的最后一道保险,我也曾这样做过[表情]  (2019-03-04 10:45) 

对的,作用就是:软件崩溃自重启!
欢迎关注微信公众号:Qt实战 (各种开源作品、经验整理、项目实战技巧,专注Qt/C++软件开发,视频监控、物联网、工业控制、嵌入式软件、国产化系统应用软件开发)QQ:517216493  WX:feiyangqingyun  QQ群:751439350
离线liuyuanan

只看该作者 9楼 发表于: 2019-04-24
我们的解决方案如下:写一个进程管理程序,这个程序使用QProcess类负责启动和关闭进程,当QProcess类监测到进程挂掉后,就重启该进程,这种方案不需要原有的进程做任何修改
离线cericking

只看该作者 10楼 发表于: 2019-05-05
回 liudianwu 的帖子
liudianwu:对的,作用就是:软件崩溃自重启! (2019-03-04 11:57) 

其实9楼的方法也很好

QProcess 对于一个进程打开和关闭是两种不同的状态,
可以每隔一段时间 如100ms,检测进程的状态 如果关闭了,则重启程序

更简洁一些 不占用网络资源
离线chamsdoncon

只看该作者 11楼 发表于: 2019-05-06
9楼的方法更适合windows系统吧,大佬这个适合跨平台
离线greensky10

只看该作者 12楼 发表于: 2019-05-14
我们也是用9楼的方法,在Linux底下用着没问题啊!
离线0011411

只看该作者 13楼 发表于: 2019-11-19
        
离线liaohanghua

只看该作者 14楼 发表于: 2020-11-25
不错 值得学习 谢谢分享
离线jobfind

只看该作者 15楼 发表于: 2021-10-18
快速回复
限100 字节
 
上一个 下一个