分享

Qt子线程的“信号队列” - Eathan的日志 - 网易博客

 lao_o 2010-01-24

Qt子线程的“信号队列”

Qt-C++   2009-11-04 16:58   阅读20   评论0  
字号:    

声明:以下纯属个人理解,看到的就当错的看...

对Qt的多线程编程没有深究,只了解了基本的用法,够我用就行了。

之所以写这篇文章是因为前几天遇到一个疑问:如果其他几个线程同时向一个线程发signal,而这个线程没有自己的事件循环,那是不是会丢失signal呢?

下面是我总结的两种子线程的工作方式

1.让子线程进入事件循环,这样的话多余的signal就会进入该线程的事件队列,不会丢失。问题是这时子线程的槽函数都是在该子线程对象所在的线程(很可能是主线程)执行,这样似乎失去了多线程的意义。
void run()
{
    exec();
}

void slot1();  //处理工作
void slot2();  //处理工作
...



2.子线程没有事件循环,直接在run里处理工作,主线程可通过信号连接到该子线程的槽来控制flag,从而控制子线程的暂停和继续。但是,如果还有另外几个线程不时地向通过slot2()给somarg赋值的话,即使给slot2()加了锁保证了不会被同时赋值,但那些同时进入的赋值信号没有队列可进,这样会不会就丢失了呢
void run()
{
    while(1)
    {
    while(flag)
    {
        dosomething(somarg);
    }
    }
}

void slot1();  //控制flag
void slot2();  //给somarg赋值
...


经试验,虽然第一种办法不需要exec()进入事件循环也可以触发槽们在调用这个线程对象的线程工作,但是这样无法保证同时传进去的信号不会丢失;加上exec()后,子线程进入事件循环,不会马上结束,并且会有事件队列,这样可以保证信号不会丢失。唯一的缺点就是这些槽不工作在子线程。

拟对策:
建立一个队列线程,CQueueThread,这个线程进入自己的事件循环,在这个类中有其它线程的对象作为成员变量,这些线程则没有各自的事件队列,直接在run中死循环工作,主线程信号连接到CQueueThread的槽来控制其他线程开始工作,这样信号会进入事件队列不会丢失,而那些死循环的繁杂工作则各自在各自的线程中运行。

这个CQueueThread可以用主线程代替(主线程必然是有事件循环的),主线程中有个槽作缓冲用,接收来自各方的命令,再鱼贯发往目标线程。

"run()开的才是新线程,QThread的构造函数以及其他成员都在你的主线程中。"

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多