前言:因为项目需要定时检查网络中设备是否能连通,需要定时去做ping操作,若是网络不通,则ping花费时间比较久(局域网大概4秒钟才能结束,当然如果设置超时时间啥的,也能很快返回,就是会报QProcess提前关闭的警告)。此外,我想要在设备连接后才去开始定时ping,要是断开后,则停止定时器,停止ping。网上有很多定时器使用线程的例子,但是没有讲到可以随时停止和开始的那种,本文结合实例,使用线程去执行定时器的定时操作。
一、定义定时器和线程变量。
在头文件中定义
#include #include #include #include QTimer *detectNetCablePulloutTimer = NULL; QThread *detectNetCablePulloutTimerThread = NULL; void startDetectNetCablePulloutTimer(); void stopDetectNetCablePulloutTimer();private slots: void checkDeviceIsOnline();
二、开始定时器、停止定时器以及销毁定时器代码。
开始定时器代码:
void CurrentXXXClass::startDetectNetCablePulloutTimer(){ // 先停止,确保开始之前是停止状态 stopDetectNetCablePulloutTimer(); if (detectNetCablePulloutTimer == NULL) { detectNetCablePulloutTimerThread = new QThread(this); detectNetCablePulloutTimer = new QTimer(); // 5秒执行一次,局域网断开时,一般耗时4-5秒 detectNetCablePulloutTimer->setInterval(5000);// 定时操作放在线程中执行 detectNetCablePulloutTimer->moveToThread(detectNetCablePulloutTimerThread); // 定时任务放在线程中执行, 必须指定参数-Qt::DirectConnection connect(detectNetCablePulloutTimer, &QTimer::timeout, this, &CurrentXXXClass::checkDeviceIsOnline, Qt::DirectConnection); void (QTimer::*pStartFun)() = &QTimer::start; // 线程开始时,调用定时器的开始 connect(detectNetCablePulloutTimerThread, &QThread::started, detectNetCablePulloutTimer, pStartFun); // 线程结束时,调用定时器的结束,所以要停止定时器,只要调用线程的quit方法, // 同时防止提示:定时器停止和开始不在同一线程错误 connect(detectNetCablePulloutTimerThread, &QThread::finished, detectNetCablePulloutTimer, &QTimer::stop); } detectNetCablePulloutTimerThread->start();}
停止定时器代码:
void CurrentXXXClass::stopDetectNetCablePulloutTimer(){ if (detectNetCablePulloutTimerThread != NULL) { // 会调用定时器的stop方法 detectNetCablePulloutTimerThread->quit(); }}
销毁定时器,在析构函数中执行:
//销毁指针变量-注意先后顺序 if (detectNetCablePulloutTimerThread != NULL) { detectNetCablePulloutTimerThread->quit(); detectNetCablePulloutTimerThread->wait(); delete detectNetCablePulloutTimerThread; detectNetCablePulloutTimerThread = nullptr; } if (detectNetCablePulloutTimer != NULL) { delete detectNetCablePulloutTimer; detectNetCablePulloutTimer = nullptr; }
三、在槽函数中执行耗时操作。
此槽函数是在子线程中执行,若是执行完后要回到主线程执行某操作,则必须通过发送一个信号方式才能达到效果。
void CurrentXXXClass::checkDeviceIsOnline(){QString ip = "192.168.1.1"; // 若是timer使用了线程,则此函数在子线程中执行,执行比较耗时的操作QProcess pingProcess;// 不设置-w参数 QString strArg = "ping " + ip + " -n 1 -i 2";// qt 6 开始命令的方法 pingProcess.startCommand(strArg, QIODevice::ReadOnly);// 不带等待时间参数 pingProcess.waitForReadyRead();// 不带等待时间参数 pingProcess.waitForFinished(); QString p_stdout = QString::fromLocal8Bit(pingProcess.readAllStandardOutput()); bool bPingSuccess = false; // 通过特殊字符串进行判断ping是否成功 if (p_stdout.contains("TTL=")) {// 网络连通 bPingSuccess = true; } else {// 网络不通 bPingSuccess = false; }}