没有任何人敢保证自己写的程序没有任何BUG,尤其是在商业项目中,程序量越大,复杂度越高,出错的概率越大,尤其是现场环境千差万别,和当初本地电脑测试环境很可能不一样,有很多特殊情况没有考虑到,如果需要保证程序7*24小时运行,则需要想一些办法能够让程序死了能够活过来,在嵌入式linux上,大部分会采用看门狗的形式来处理,程序打开看门狗驱动后,定时喂狗,一旦超过规定的时间,则硬件软复位等。这种方式相对来说比较可靠,如果需要在普通PC机上运行怎办呢?本篇文章提供一个软件实现守护进程的办法,原理就是udp通信,单独写个守护进程程序,专门负责检测主程序是否存在,不存在则启动。主程序只需要启动live类监听端口,收到hello就回复ok就行。
为了使得兼容任意程序,特意提炼出来共性,增加了多种设置。
1:可设置检测的程序名称。
2:可设置udp通信端口。
3:可设置超时次数。
4:自动记录已重启次数。
5:自动记录最后一次重启时间。
6:是否需要重新刷新桌面。
7:可重置当前重启次数和最后重启时间。
8:自动隐藏的托盘运行或者后台运行。
9:提供界面设置程序名称已经开启和暂停服务。
完整代码下载:https://download.csdn.net/download/feiyangqingyun/10989964



守护进行核心代码:

#pragma execution_character_set("utf-8")
#include "frmmain.h"
#include "ui_frmmain.h"
#include "qtimer.h"
#include "qudpsocket.h"
#include "qsharedmemory.h"
#include "qprocess.h"
#include "qdatetime.h"
#include "qapplication.h"
#include "qdesktopservices.h"
#include "qmessagebox.h"
#if (QT_VERSION > QT_VERSION_CHECK(5,0,0))
#include "qstandardpaths.h"
#endif#include "app.h"frmMain::frmMain(QWidget *parent) : QWidget(parent), ui(new Ui::frmMain)
{ui->setupUi(this);this->initForm();
}frmMain::~frmMain()
{delete ui;
}void frmMain::changeEvent(QEvent *event)
{//隐藏当前界面,最小化到托盘if(event->type() == QEvent::WindowStateChange) {if(windowState() & Qt::WindowMinimized) {hide();}}QWidget::changeEvent(event);
}void frmMain::initForm()
{count = 0;ok = false;//每秒钟定时询问心跳timerHeart = new QTimer(this);timerHeart->setInterval(2000);connect(timerHeart, SIGNAL(timeout()), this, SLOT(sendHearData()));//从6050端口开始,如果绑定失败则将端口加1,直到绑定成功udp = new QUdpSocket(this);int port = 6050;while(!udp->bind(port)) {port++;}connect(udp, SIGNAL(readyRead()), this, SLOT(readData()));if (App::TargetAppName.isEmpty()) {ui->btnStart->setText("启动");ui->btnStart->setEnabled(false);timerHeart->stop();} else {ui->btnStart->setText("暂停");ui->btnStart->setEnabled(true);timerHeart->start();}ui->txtAppName->setText(App::TargetAppName);ui->txtAppName->setFocus();
}void frmMain::sendHearData()
{udp->writeDatagram("hello", QHostAddress::LocalHost, App::TargetAppPort);//判断当前是否没有回复if (!ok) {count++;} else {count = 0;ok = false;}//如果超过规定次数没有收到心跳回复,则超时重启if (count >= App::TimeoutCount) {timerHeart->stop();QSharedMemory mem(App::TargetAppName);if (!mem.create(1)) {killApp();}QTimer::singleShot(1000 , this, SLOT(killOther()));QTimer::singleShot(3000 , this, SLOT(startApp()));QTimer::singleShot(4000 , this, SLOT(startExplorer()));}
}void frmMain::killApp()
{QProcess *p = new QProcess;p->start(QString("taskkill /im %1.exe /f").arg(App::TargetAppName));
}void frmMain::killOther()
{QProcess *p = new QProcess;p->start(QString("taskkill /im %1.exe /f").arg("WerFault"));//重建缓存,彻底清除托盘图标if (App::ReStartExplorer) {QProcess *p1 = new QProcess;p1->start("taskkill /f /im explorer.exe");}
}void frmMain::startApp()
{if (ui->btnStart->text() == "开始" || ui->btnStart->text() == "启动") {count = 0;return;}QProcess *p = new QProcess;p->start(QString("\"%1/%2.exe\"").arg(qApp->applicationDirPath()).arg(App::TargetAppName));count = 0;ok = true;timerHeart->start();App::ReStartCount++;App::ReStartLastTime = QDateTime::currentDateTime().toString("yyyy-MM-dd HH:mm:ss");App::writeConfig();ui->labCount->setText(QString("已重启 %1 次").arg(App::ReStartCount));ui->labInfo->setText(QString("最后一次重启在 %1").arg(App::ReStartLastTime));
}void frmMain::startExplorer()
{//取得操作系统目录路径,指定操作系统目录下的explorer程序,采用绝对路径,否则在64位操作系统下无效
#if (QT_VERSION > QT_VERSION_CHECK(5,0,0))QString str = QStandardPaths::writableLocation(QStandardPaths::ApplicationsLocation);
#elseQString str = QDesktopServices::storageLocation(QDesktopServices::ApplicationsLocation);
#endifif (App::ReStartExplorer) {str = QString("%1\\Windows\\explorer.exe").arg(str.mid(0, 2));QProcess *p = new QProcess(this);p->start(str);}
}void frmMain::readData()
{QByteArray tempData;do {tempData.resize(udp->pendingDatagramSize());udp->readDatagram(tempData.data(), tempData.size());QString data = QLatin1String(tempData);if (data.right(2) == "OK") {count = 0;ok = true;}} while (udp->hasPendingDatagrams());
}void frmMain::on_btnOk_clicked()
{App::TargetAppName = ui->txtAppName->text();if (App::TargetAppName == "") {QMessageBox::critical(this, "提示", "应用程序名称不能为空!");ui->txtAppName->setFocus();return;}App::writeConfig();ui->btnStart->setEnabled(true);
}void frmMain::on_btnStart_clicked()
{count = 0;if (ui->btnStart->text() == "暂停") {timerHeart->stop();ui->btnStart->setText("开始");} else {timerHeart->start();ui->btnStart->setText("暂停");}
}void frmMain::on_btnReset_clicked()
{App::ReStartCount = 0;App::ReStartLastTime = "2019-01-01 12:00:00";App::writeConfig();ui->txtAppName->setText(App::TargetAppName);ui->labCount->setText(QString("已重启 %1 次").arg(App::ReStartCount));ui->labInfo->setText(QString("最后一次重启在 %1").arg(App::ReStartLastTime));QMessageBox::information(this, "提示", "重置配置文件成功!");
}

使用主程序核心代码:

#include "applive.h"
#include "qmutex.h"
#include "qudpsocket.h"
#include "qstringlist.h"
#include "qapplication.h"
#include "qdatetime.h"
#include "qdebug.h"#define TIMEMS qPrintable(QTime::currentTime().toString("HH:mm:ss zzz"))QScopedPointer<AppLive> AppLive::self;
AppLive *AppLive::Instance()
{if (self.isNull()) {QMutex mutex;QMutexLocker locker(&mutex);if (self.isNull()) {self.reset(new AppLive);}}return self.data();
}AppLive::AppLive(QObject *parent) : QObject(parent)
{udpServer  = new QUdpSocket(this);QString name = qApp->applicationFilePath();QStringList list = name.split("/");appName = list.at(list.count() - 1).split(".").at(0);
}void AppLive::readData()
{QByteArray tempData;do {tempData.resize(udpServer->pendingDatagramSize());QHostAddress sender;quint16 senderPort;udpServer->readDatagram(tempData.data(), tempData.size(), &sender, &senderPort);QString data = QLatin1String(tempData);if (data == "hello") {udpServer->writeDatagram(QString("%1OK").arg(appName).toLatin1(), sender, senderPort);}} while (udpServer->hasPendingDatagrams());
}bool AppLive::start(int port)
{bool ok = udpServer->bind(port);if (ok) {connect(udpServer, SIGNAL(readyRead()), this, SLOT(readData()));qDebug() << TIMEMS << "Start AppLive Ok";}return ok;
}void AppLive::stop()
{udpServer->abort();disconnect(udpServer, SIGNAL(readyRead()), this, SLOT(readData()));
}

Qt编写守护程序保证程序一直运行(开源)相关推荐

  1. 【转】Qt编写串口通信程序全程图文讲解

    Qt编写串口通信程序全程图文讲解 本文章原创于www.yafeilinux.com 转载请注明出处. (说明:我们的编程环境是windows xp下,在Qt Creator中进行,如果在Linux下或 ...

  2. 用Qt 编写黑白棋程序

    用Qt 编写黑白棋程序 前言:这是继三天学习c++后,学习了两天Qt 做的项目. 软件设计开发流程: 瀑布模型: 可行性分析 时间: 共8天(其中前两天学习Qt3.1 与Qt4.3.5,中间编程,最后 ...

  3. Windows下Qt编写Android应用程序的环境搭建记录

    目录 前言 配置开发环境 安装JDK1.8 安装SDK 安装NDK Qt配置OpenCV C++&Java混合编程 结语 前言 由于要在手机上实现一些应用,例如部署目标检测模型.添加单目测距定 ...

  4. qt编写的android程序在android系统中运行,中文显示乱码的解决方法

    在main函数中添加 QTextCodec::setCodecForTr(QTextCodec::codecForLocale()); QTextCodec::setCodecForTr(QTextC ...

  5. Qt编写串口通信程序全程图文讲解

    (说明:我们的编程环境是windows xp下,在Qt Creator中进行,如果在Linux下或直接用源码编写,程序稍有不同,请自己改动.) 在Qt中并没有特定的串口控制类,现在大部分人使用的是第三 ...

  6. 编写的Matlab程序在运行一段时间后速度会变慢

    我们用matlab编写程序进行数据运算的时候经常会遇到这种情况,就是编写完立马运行,运行速度会非常快,我们感到很满意.但是运行一段时间后,速度就不是那么理想了,恨不得推他一把.这是什么原因呢?通过一段 ...

  7. Qt编写密钥生成器+使用demo(开源)

    在很多商业软件中,需要提供一些可以试运行的版本,这样就需要配套密钥机制来控制,纵观大部分的试用版软件,基本上采用以下几种机制来控制. 1:远程联网激活,每次启动都联网查看使用时间等,这种方法最完美,缺 ...

  8. Qt编写GIF录屏工具(开源)

    在平时的写作过程中,经常需要将一些操作动作和效果图截图成gif格式,使得涵盖的信息更全面更生动,有时候可以将整个操作过程和运行效果录制成MP4,但是文件体积比较大,而且很多网站不便于上传,基本上都支持 ...

  9. Qt编写的开源帖子集合(懒人专用)

    回顾自己学习Qt以来九年了,在这九年多时间里面,从本论坛学习不到不少的东西,今天特意整了一下自己开源过的资源的帖子,整理一起方便大家直接跳转下载,不统计不知道,一统计吓一跳,不知不觉开源了这么多代码, ...

最新文章

  1. (每日一题)P3312 [SDOI2014]数表(经典莫比乌斯反演 + 树状数组维护离线询问)
  2. [AaronYang]C#人爱学不学8[事件和.net4.5的弱事件深入浅出]
  3. 第五周实践项目7 后缀表达式
  4. 支付宝异步回调返回success_深入解决异步编程Promise对象的学习
  5. C++基础知识(三)—— 常量
  6. 月薪2W和月薪10W的差别,怎么判断一个产品经理的专业水平高低?
  7. easy Connect连接成功,但是虚拟IP地址获取失败,而导致网页无法打开
  8. jetbrains webstorm汉化包
  9. MCSA / Windows Server 2016 系列笔记 -开篇
  10. Cannot read property ‘detail‘ of undefined点击详情没反应
  11. Entersekt欢迎Nicolas Huss加入董事会
  12. OPNET计算机网络仿真 实验作业3 - 标准应用介绍
  13. 推荐61个免费最好软件
  14. 〖Python接口自动化测试实战篇⑥〗- 接口抓包工具 Chrome 的使用
  15. 计算机上无法获取,电脑无法获取有效ip 修复ip地址配置方法
  16. H3C室外无线AP(WA4320X)胖瘦切换设置方法
  17. HDU 5238 Calculator(中国剩余定理+线段树)
  18. 自动化测试平台化[v1.0.0][自动化测试基本需求]
  19. 一个简单的Python樱花飘落动画代码
  20. APP测试面试题汇总,面试必考一定要看

热门文章

  1. yy欢聚时代软件测试笔试题
  2. 软考和PMP哪个更好考?
  3. 算法题解——将字符串转化为ip地址
  4. Python | OpenCV画图显示为一个全黑画布的情况
  5. [附源码]java毕业设计社区生鲜电商平台
  6. 利用pe系统重装电脑
  7. K8S集群中Pod与Pod之间网络故障排查思路
  8. 你心心念念的 GitHub手机版APP终于来咯
  9. html5 canvas创建弹性碰撞动画
  10. 使用动态规划弹性扩容机器。使用拉链法O(N)求所有集合