Qt 定时器放在线程中执行,支持随时开始和停止定时器。

前言:因为项目需要定时检查网络中设备是否能连通,需要定时去做ping操作,若是网络不通,则ping花费时间比较久(局域网大概4秒钟才能结束,当然如果设置超时时间啥的,也能很快返回,就是会报QProcess提前关闭的警告)。此外,我想要在设备连接后才去开始定时ping,要是断开后,则停止定时器,停止ping。网上有很多定时器使用线程的例子,但是没有讲到可以随时停止和开始的那种,本文结合实例,使用线程去执行定时器的定时操作。

一、定义定时器和线程变量。

在头文件中定义

    #include 
    #include 
    #include 
    #include 

    /** 监听网线拔出或者wifi断开情况的定时器 ***/
    QTimer *detectNetCablePulloutTimer = NULL;
    /** 执行定时器的线程 **/
    QThread *detectNetCablePulloutTimerThread = NULL;

    /** 开始检测网线拔出时或者wifi断开时的定时器 **/
    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;
    }
}

你可能感兴趣的:(C++,Windows,qt,开发语言)