Qt项目实例_03_Qt多线程与定时器QTimer(信号槽机制实现数据共享;解决Timers cannot be started from another thread问题;推荐使用信号启动定时器)

在进行程序编写中,我们有时会有在子线程中使用定时器不间断运行并且将子线程产生的数据发送给主线程的需求,在实际的编写过程中可能会碰到QObject::startTimer: Timers cannot be started from another thread的问题,本文将会介绍产生原因及解决方法。

1. 错误用法及原因

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

在这里插入图片描述
接下来在主线程中创建该线程对象,并调用它的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线程启动后创建定时器。
在这里插入图片描述
有些地方需要注意:

  • 1.不能像下面这样给定时器指定父对象
m_pTimer =newQTimer(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线程中,否则会出错。

在这里插入图片描述
上面的代码将出现以下错误:

QObject::killTimer: Timers cannot be stopped from another thread

综上,子类化线程类的方法可行,但是不太好。

2. 正确用法二(推荐使用)

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

在这里插入图片描述
通过moveToThread()方法改变定时器所处的线程,不要给定时器设置父类,否则该函数将不会生效。

在信号槽连接时,我们增加了一个参数——连接类型,先看看该参数可以有哪些值:

  • Qt::AutoConnection:默认值。如果接收者处于发出信号的线程中,则使用Qt::DirectConnection,否则使用Qt::QueuedConnection,连接类型由发出的信号决定。
  • Qt::DirectConnection:信号发出后立即调用槽函数,槽函数在发出信号的线程中执行。
  • Qt::QueuedConnection:当控制权返还给接收者信号的事件循环中时,开始调用槽函数。槽函数在接收者的线程中执行。

回到我们的例子,首先将定时器所处的线程改为新建的线程,然后连接信号槽,槽函数在定时器所处的线程中执行。

参考:https://www.jb51.net/article/234607.htm;https://www.cnblogs.com/ybqjymy/p/18045131

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

十月旧城

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值