【快速上手QT】04-定时器Timer

本文介绍了如何在Qt中使用定时器,包括startTimer的基本用法,QTimer类的信号与槽机制,以及如何通过timerEvent处理定时器触发。还探讨了非周期计数的singleShot函数。
摘要由CSDN通过智能技术生成

先来个小示例

我们先简单的来触发一下定时器。

#include "Zhetu.h"

#include <qdebug.h>

void Zhetu::timerEvent(QTimerEvent* event) {	//定时器触发函数
	qDebug() << "Hello world";
}

Zhetu::Zhetu(QWidget *parent): QMainWindow(parent){
	this->setFixedSize(500, 500);

	this->startTimer(1000);			//开启以1000ms为周期的定时器
}

上面的代码很简单,就是开启一个1000ms的定时器,定时器触发的时候调用timerEvent这个函数,并且输出调试信息“Hello world”。

这边有几个需要注意的点,就是定时器触发函数是固定的,就是要下面这个形式。

void timerEvent(QTimerEvent* event)

当然,形参的名字可以自己起,但是返回值,函数名,形参类型都是需要固定的。

并且我们能够看得出来,开启定时器的对象是我们的主界面(在主界面的构造函数中)。因此我们的定时器触发函数得是我们主界面这个类的成员函数。

startTimer

我们使用startTimer这个函数就使得我们的主界面调用了定时器,那么我是怎么知道用这个函数的呢?答案就是QT助手。

我们的主界面类继承的是QMainWindow这个类,那么我们就去搜索QMainWindow,我们去找有没有定时器相关的函数,答案是没有的。

既然QMainWindow没有,那么我们就去它的父类去找。

然而我们在它的父类里也找不到,那么我们再从QWidget的父类中去寻找。

QWidget的父类有两个,分别是QObject和QPaintDevice。

凭借我高超的英语水平(四级差122分),一眼就能看出我们需要从QObject里找,因为QPaintDevice根据直译来判断,应该是和图形化相关的。

果然我们能在QObject中找到相关函数。

当我们点击进去之后就能看到关于这个函数的详细的解释了。

凭借我高超的英语水平(四级差122分),一眼就能看出我看不懂,所以我们稍微翻译参考一下。

根据这个粗糙的翻译,我们可以知道调用这个函数之后会返回一个标识符(ID),并且定时器事件会在每个时间间隔触发,直到我们调用killTimer这个函数,很明显这个函数是用来杀死(kill)定时器事件的。

我们再次点击进入killTimer这个函数。

我们知道startTimer会返回一个定时器ID,那么我们取消定时器,使用killTimer用的参数也是这个ID。

再回到我们的startTimer函数的解释里。

定时器事件触发之后,会调用TimerEvent函数。如果有多个定时器函数,我们还可以通过TimerEvent的参数去调用timerId来获取当前触发定时器函数的定时器ID是什么。

综上我们可以知道以下几个关键的信息。

  • startTimer是QObject的函数,而大多数组件都直接或间接继承了QObject,所以大多数组件都可以开启定时器。
  •  我们可以通过调用startTimer获取的定时器ID来删除定时器事件或是判断触发定时器事件的是哪个ID
  • 同类组件哪怕开启了多个定时器,但是定时器触发函数是固定的一个。

这样我们大概知道了怎么通过OBject的成员函数来开启定时器事件之后,我们就可以来测试一下了。

#include "Zhetu.h"

#include <qdebug.h>
#include <qpushbutton.h>

int windowID, buttonID;

void Zhetu::timerEvent(QTimerEvent* event) {	//定时器触发函数
	static int count = 0;
	if (event->timerId() == windowID) {		//获取触发定时器的ID,比较是否是主界面的定时器ID
		qDebug() << "w";
		if (++count >= 10) {
			killTimer(windowID);			//取消主界面定时器
			qDebug() << "kill window timer";
		}
	}
}

void QAbstractButton::timerEvent(QTimerEvent* event){
	static int count = 0;
	if (event->timerId() == buttonID) {		//获取触发定时器的ID,比较是否是按键的定时器ID
		qDebug() << "b";
		if (++count >= 10) {
			killTimer(buttonID);			//取消按键定时器
			qDebug() << "kill button timer";
		}
	}
}

Zhetu::Zhetu(QWidget *parent): QMainWindow(parent){
	this->setFixedSize(500, 500);

	QPushButton* button = new QPushButton("Timer", this);
	buttonID = button->startTimer(500);			//开启以500ms为周期的定时器

	windowID = this->startTimer(1000);			//开启以1000ms为周期的定时器
}

上面代码中,我用主界面开启了一个定时器,用一个普通的按键也开启了一个定时器。

由于它们所属的类不一样,因此需要重写两次TimerEvent函数。并且它们所属的类需要区分来开。

主界面的定时器触发函数的所属类就写Zhetu(我写的主界面类的名称)。

而按键的定时器触发函数的所属类需要写QAbstractButton,如果写QPushButton的话是不允许的,我已经试过了。

在各自的定时器触发函数中我通过调用参数的timerId来获取触发定时器事件的ID进行定时器ID的判断,各自触发十次之后取消定时器事件。

从下面运行结果的截图中也可以看的出来是没问题的。

QTimer

上面的开启定时器其实是够用了,不过这里还是介绍一下QTimer这个类,因为这个类才算得上是真正的定时器。

与startTimer不同的是,Timer开启定时器的方式是信号与槽函数。

我们照例先来个小例子。

#include "Zhetu.h"

#include <qdebug.h>
#include <QTimer> 

Zhetu::Zhetu(QWidget *parent): QMainWindow(parent){
	this->setFixedSize(500, 500);

	QTimer* t1 = new QTimer(this);
	connect(t1, &QTimer::timeout, [&]() {
		qDebug() << "hello world";
		});
	t1->start(1000);
}

既然触发定时器事件的是信号和槽函数,那么我们先看看QTimer都有些什么信号和槽函数。

看得出来信号只有一个,那就是timeout,凭借我高超的英语水平(四级差122分),一眼就能看出这个信号就是定时器定的时间到了之后发出的信号,也就是是计数溢出信号。

我们将这个信号和自定义的定时器触发函数进行绑定(connect),就可以达到触发定时器事件的目的了。

而我们可以调动槽函数去控制这个定时器 。

很明显start就是开启定时器,而stop就是关闭定时器。

那么我们通过上面一个小例子就可以知道如何使用Timer来实现定时器的周期计数了。

这边再介绍一个非周期计数,也就是只触发一次定时器事件的函数。

#include "Zhetu.h"

#include <qdebug.h>
#include <QTimer> 

Zhetu::Zhetu(QWidget *parent): QMainWindow(parent){
	this->setFixedSize(500, 500);

	QTimer* t1 = new QTimer(this);
	t1->singleShot(1000, []() {
		qDebug() << "Hello";
		});
}

Zhetu::~Zhetu()
{}

虽然我们使用周期定时器也可以达到同样的效果(触发一次之后就关闭周期定时器),但是这个非周期的定时器有一个特点,那就是它属于静态成员函数,也就是说我们不需要创建QTimer对象就可以使用这个函数。

这也是这个函数比较方便的一点。

QTimer::singleShot(1000, []() {
		qDebug() << "helo world";
		});

小结

关于定时器的使用我这里只是介绍了一点,但是对于我们需要快速上手的小伙伴来说是够用了,想要更近一步的小伙伴可以自己去查阅资料以及QT助手去自主进阶学习。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值