文章目录

  • 一、定时器QTimer类
  • 二、在多线程中使用QTimer
    • 1.错误用法
    • 2.正确用法一
      • 1) 不能像下面这样给定时器指定父对象
      • 2) 必须要加上事件循环exec()
    • 3.正确用法二
  • 三、案例

参考:https://www.cnblogs.com/hellovenus/p/qt_thread_timer.html

一、定时器QTimer类

The QTimer class provides repetitive and single-shot timers.
The QTimer class provides a high-level programming interface for timers. To use it, create a QTimer, connect its timeout() signal to the appropriate slots, and call start(). From then on, it will emit the timeout() signal at constant intervals.

上面这段话摘自Qt助手文档,我们使用QTimer类定义一个定时器,它可以不停重复,也可以只进行一次便停止。
使用起来也很简单:

QTimer *timer = new QTimer(this);
connect(timer, SIGNAL(timeout()), this, SLOT(update()));
timer->start(1000);

创建一个QTimer对象,将信号timeout()与相应的槽函数相连,然后调用start()函数。接下来,每隔一段时间,定时器便会发出一次timeout()信号。
更多用法这里就不讲了,您可以自行参考官方文档。比如如何停止、如何令定时器只运行一次等。

二、在多线程中使用QTimer

1.错误用法

您可能会这么做:
子类化QThread,在线程类中定义一个定时器,然后在run()方法中调用定时器的start()方法。

TestThread::TestThread(QObject *parent): QThread(parent)
{m_pTimer = new QTimer(this);connect(m_pTimer, &QTimer::timeout, this, &TestThread::timeoutSlot);
}void TestThread::run()
{m_pTimer->start(1000);
}void TestThread::timeoutSlot()
{qDebug() << QString::fromLocal8Bit("当前线程id:") << QThread::currentThread();
}

接下来在主线程中创建该线程对象,并调用它的start()方法:

m_pThread = new TestThread(this);
m_pThread->start();

看似十分自然,没有什么不妥,然而,编译器将通知下面的错误信息:
QObject::startTimer: Timers cannot be started from another thread
——定时器不能被其它线程start。
我们来分析一下:
刚开始只有主线程一个,TestThread的实例是在主线程中创建的,定时器在TestThread的构造函数中,所以也是在主线程中创建的。
当调用TestThread的start()方法时,这时有两个线程。定时器的start()方法是在另一个线程中,也就是TestThread中调用的。
创建和调用并不是在同一线程中,所以出现了错误。
具体的原理可参考官方文档——点我
每个QObject实例都有一个叫做“线程关系”(thread affinity)的属性,或者说,它处于某个线程中。
默认情况下,QObject处于创建它的线程中。
当QObject接收队列信号(queued signal)或者传来的事件(posted event),槽函数或事件处理器将在对象所处的线程中执行。
根据以上的原理,Qt使用计时器的线程关系(thread affinity)来决定由哪个线程发出timeout()信号。正因如此,你必须在它所处的线程中start或stop该定时器,在其它线程中启动定时器是不可能的。

2.正确用法一

在TestThread线程启动后创建定时器。
void TestThread::run()
{
m_pTimer = new QTimer();
m_pTimer->setInterval(1000);
connect(m_pTimer, &QTimer::timeout, this, &TestThread::timeoutSlot);
m_pTimer->start();
this->exec();
}
有些地方需要注意:

1) 不能像下面这样给定时器指定父对象

m_pTimer = new QTimer(this);
否则会出现以下警告:
QObject: Cannot create children for a parent that is in a different thread.
(Parent is TestThread(0x709d88), parent’s thread is QThread(0x6e8be8), current thread is TestThread(0x709d88) 
因为TestThread对象是在主线程中创建的,它的QObject子对象也必须在主线程中创建。所以不能指定父对象为TestThread。

2) 必须要加上事件循环exec()

否则线程会立即结束,并发出finished()信号。
另外还有一点需要注意,与start一样,定时器的stop也必须在TestThread线程中,否则会出错。
void TestThread::timeoutSlot()
{
m_pTimer->stop();
qDebug() << QString::fromLocal8Bit(“当前线程id:”) << QThread::currentThread();
}
上面的代码将出现以下错误:
QObject::killTimer: Timers cannot be stopped from another thread
综上,子类化线程类的方法可行,但是不太好。

3.正确用法二

无需子类化线程类,通过信号启动定时器。

TestClass::TestClass(QWidget *parent): QWidget(parent)
{m_pThread = new QThread(this);m_pTimer = new QTimer();m_pTimer->moveToThread(m_pThread);m_pTimer->setInterval(1000);connect(m_pThread, SIGNAL(started()), m_pTimer, SLOT(start()));connect(m_pTimer, &QTimer::timeout, this, &ThreadTest::timeOutSlot, Qt::DirectConnection);
}

通过moveToThread()方法改变定时器所处的线程,不要给定时器设置父类,否则该函数将不会生效。
在信号槽连接时,我们增加了一个参数——连接类型,先看看该参数可以有哪些值:
● Qt::AutoConnection:默认值。如果接收者处于发出信号的线程中,则使用Qt::DirectConnection,否则使用Qt::QueuedConnection,连接类型由发出的信号决定。
● Qt::DirectConnection:信号发出后立即调用槽函数,槽函数在发出信号的线程中执行。
● Qt::QueuedConnection:当控制权返还给接收者信号的事件循环中时,开始调用槽函数。槽函数在接收者的线程中执行。
回到我们的例子,首先将定时器所处的线程改为新建的线程,然后连接信号槽,槽函数在定时器所处的线程中执行。


三、案例

#include <QtCore>int threadCount = 4;void SubThread() {for (int var = 0; var < threadCount; ++var) {int *count = new int(rand() % 30);qDebug() << *count;QTimer *timer = new QTimer;timer->setInterval (500);QThread *thread = new QThread();thread->setObjectName (QString::number (var));timer->moveToThread (thread);QObject::connect (timer,&QTimer::timeout,[=]{qDebug() << QThread::currentThread () << (*count)--;if(*count < 0 ) {timer->stop ();thread->quit ();//                thread->deleteLater ();threadCount--;}});QObject::connect(thread,&QThread::started,timer, QOverload<>::of( &QTimer::start ) );QObject::connect(thread,&QThread::finished ,[]{qDebug() << QThread::currentThread () << "Stopped!";} );QObject::connect(thread,&QThread::destroyed,[]{qDebug() << QThread::currentThread () << "destroyed!";} );thread->start ();}
}int main(int argc, char *argv[])
{QCoreApplication a(argc, argv);QTimer tm;tm.start (500);QObject::connect (&tm,&QTimer::timeout, [&]{        // 定时监测子线程数量static int count = 0;if( count == 0 ){SubThread();}if( count != threadCount) {count = threadCount;qDebug() << QThread::currentThread () << "main thread ---------->" << threadCount;}if(threadCount == 0) {tm.stop ();qApp->quit ();}} );a.exec();return 0;
}

[多线程]多线程使用QTimer相关推荐

  1. Java多线程——多线程的基本概念和使用

    一.进程和线程的基础知识 1.进程和线程的概念 进程:运行中的应用程序称为进程,拥有系统资源(cpu.内存) 线程:进程中的一段代码,一个进程中可以有多段代码.本身不拥有资源(共享所在进程的资源) 在 ...

  2. java多线程--多线程基础小结

    什么是线程? 在同一个进程中可以执行多个任务,每一个任务可以看做一个线程. 线程是程序的执行单元,执行路径,使程序使用cpu的最基本单位 一个进程如果只有一条执行路径,那么就是单线程的 一个进程如果有 ...

  3. java多线程中方法_java中多线程 - 多线程中的基本方法

    介绍一下线程中基本的方法使用 线程睡眠sleep() Thread.sleep(毫秒);我们可以通过sleep方法设置让线程睡眠.可以看到sleep是个静态方法 public static nativ ...

  4. java 线程的基本概念_Java多线程——多线程的基本概念和使用

    一.进程和线程的基础知识 1.进程和线程的概念 进程:运行中的应用程序称为进程,拥有系统资源(cpu.内存) 线程:进程中的一段代码,一个进程中可以有多段代码.本身不拥有资源(共享所在进程的资源) 在 ...

  5. java线程知识梳理_Java多线程——多线程相关知识的逻辑关系梳理

    1 学习多线程知识的根本目标 多线程知识的根本目标是:设计稳健的并发程序. 当然,本文无法回答这个实践性很强的问题(这与具体的业务相关,涉及到具体的策略),本文主要阐述相关知识之间的关系,希望初学者不 ...

  6. java中多线程 - 多线程中的基本方法

    介绍一下线程中基本的方法使用 线程睡眠sleep() Thread.sleep(毫秒);我们可以通过sleep方法设置让线程睡眠.可以看到sleep是个静态方法 public static nativ ...

  7. android 单元测试 多线程,多线程之单元测试(Junit)

    多线程测试服务 1.新建一个核心数为100 的线程池 ExecutorService service = Executors.newFixedThreadPool(100); 2.执行一个阻塞不大的任 ...

  8. oracle游标多线程,多线程jdbc游标分页查询原理 oracle和mysql分页

    分页查询中如何使用多线程加快处理速度? 分页sql是拼起来的现在怎么优化可以提高展示速度:优化思路: 1.试试并发多线程访问,然后把多线程获取的结果合并在一起. 2.做索引,加快查询速度. 3.把经常 ...

  9. 多线程——多线程同步的三种实现方法

    当使用多线程访问同一个资源的时候,非常容易出现线程安全的问题(例如,当多个线程同时对一个数据进行修改的时候,会导致某些线程对数据的修改丢失). 因此,需要采用同步机制来解决这种问题.而Java主要提供 ...

最新文章

  1. 现在企业会要求JAVA人员会怎样的技术呢?
  2. 有没有大神知道国产加密算法SM2的详细介绍
  3. 系统重装助手教你如何在Microsoft Edge中恢复“关闭所有选项卡”警告
  4. 【python】r+,w+ 全局变量
  5. 项目方案-标书-文档等等编写规范
  6. php +html5 websocket 聊天室
  7. Python撰写mail
  8. SOA企业需要考虑的事项
  9. Memcached的几种Java客户端(待实践)
  10. 黑客技术之加密你的磁盘
  11. debian下,下载linux内核
  12. java string 转 class_java-String类的转换功能
  13. Java练手项目(好玩又有趣)
  14. CentOS或Linux中,查看Tomcat版本的三种方式
  15. 简易抽奖软件逻辑实现
  16. html5图片任何改不透明度,如何巧改插入图片的透明度
  17. 青年教师计算机培训美篇,小学生停课不停学美篇 小学生停课不停学美篇开头语...
  18. simpleDateFormat 和 TimeZone
  19. 我的世界服务器物品上锁指令,如何用命令给“箱子”上锁?我的世界:这不是愚人节玩笑!...
  20. 解决--cidaemon cpu 100%

热门文章

  1. [转]超级强大的SVG SMIL animation动画详解
  2. linux驱动调试技巧:灌寄存器---------以mma7660为例
  3. FastAdmin表单工具栏自定义按钮
  4. MySql命令行基本语法
  5. org.jasypt.exceptions.EncryptionOperationNotPossibleException: null
  6. Cisco(34)——BGP的十三条选路原则
  7. 软件测试工程师华为面经
  8. 基频和倍频的概念_一倍频分析
  9. 2021-08-12:浏览器使用window.open 新页签被拦截的问题
  10. Java后端学Android(13)-使用网络技术