Qthread用法详解

原文

http://blog.sina.com.cn/s/blog_7c05cb370101bprl.html


我在线程类(继承QThread)里面用了一个QTimer,并有一个对应的slot函数。

运行的过程中会报错:
QObject: Cannot create children for a parent that is in a different thread.
(Parent is myThread(0x98f6e10), parent's thread is QThread(0x98d08a0), current thread is myThread(0x98f6e10)

为什么呢?

因为QThread的对象依附在主线程中,所以他的slot函数会在主线程中执行,而不是次线程。除非:
  • QThread 对象依附到次线程中(通过movetoThread)
  • slot 和信号是直接连接,且信号在次线程中发射


然后通过查找资料,可以得到:

如果没有对qthread中run()函数进行重载,那么run()的动作就是简单的exec(),当调用start()之后,qthread进入它的事件循环当中去,run()函数便是事件循环的起始点。常见的事件循环的表现形式为在run()里面写一个死循环,例如:

run(){while(1) {you_do_something_here();}}

但有时候我们希望能有一个时钟作为事件驱动,这个时候我们引入qtimer,因此不再重载run(),同时因为qthread对象是存在于创建它的线程当中的,为了能让qtimer真正驱动qthread中的某个函数,我们需要调用moveToThread()这个函数,把qthread对象移动到它自己对应的线程当中去。

参考:http://fateboat.72pines.com/2009/12/24/qthread中使用qtimer/



最后我把run函数注释掉不去重载它,把timer初始化放到Myhread的构造函数里面,运行通过,没有报错。


我在MyThread的构造函数里面添加了一句:

 

qDebug() << "timer: " << timer->thread();
输出结果为:

 

timer: QThread(0x8b98388) 
恩,明白了,如果不重载QThread的run函数,那么子线程类( MyThread)构造函数new的对象和slot都在主线程中运行。
但如果是这样子的话?那还算多线程吗?


第二种方法:在run中发个信号到主线程中槽函数,由于你的槽函数是在主线程中运行,就不会出现这种现象了(没有尝试,感觉体现不了多线程的东西

第三种方法:其实,这个方法太简单,太好用了。定义一个普通的QObject派生类,然后将其对象move到QThread中。使用信号和槽时根本不用考虑多线程的存在。也不用使用QMutex来进行同步,Qt的事件循环会自己自动处理好这个。

(这种方法来自http://hi.baidu.com/cyclone/blog/item/a33794ee00acba262cf53442.html#0

 

 
   
 #include <</SPAN>QtCore/QCoreApplication> 
#include <</SPAN>QtCore/QObject>
#include <</SPAN>QtCore/QThread> #include <</SPAN>QtCore/QDebug>
class Dummy:public QObject
{
Q_OBJECT public:
Dummy(QObject* parent=0):QObject(parent) {}
public slots:
void emitsig()
{
emit sig();
}
signals:
void sig();
};
class Object:public QObject { Q_OBJECT
public:
Object(){}
public slots: void slot() { qDebug()<<"from thread slot:" <<QThread::currentThreadId(); }
};
#include "main.moc"
int main(int argc, char *argv[])
{ QCoreApplication a(argc, argv); qDebug()<<"main thread:"<<QThread::currentThreadId(); QThread thread;
Object obj; Dummy dummy; obj.moveToThread(&thread);
QObject::connect(&dummy, SIGNAL(sig()), &obj, SLOT(slot())); thread.start();
dummy.emitsig(); return a.exec();
}
 
   结果:恩,slot确实不在主线程中运行(这么简单不值得欢呼么?) 
   

 

 
   
main thread: 0x1a5c 
from thread slot: 0x186c
 
   

感觉前2种方法有点取巧,第3种貌似体现了多线程,但是好像又不太好(PS:对多线程理解有限,待提高)

个人的问题是:

1.个人感觉不重载run函数,都体现不了多线程的东西;

2.如果有程序必须重载run函数,再遇到这个问题如何解决?

3.如果时间要求精确,不可以用QTimer呢?那用什么代替呢?

这些都是要面对并必须解决的问题。

PS:如果在run里面使用了

 

timer->start();

就必须在run函数的最后面添加

 

exec();

以使线程可以处理循环event



mark一下。继续研究~








你可能感兴趣的:(qt,QThread)