原文
http://blog.sina.com.cn/s/blog_7c05cb370101bprl.html
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 对象依附到次线程中(通过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
#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一下。继续研究~