• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            zhonghua

            C++博客 首頁 新隨筆 聯系 聚合 管理
              72 Posts :: 1 Stories :: 4 Comments :: 0 Trackbacks

                    昨晚在一個郵件列表里面看見一個關于在線程種使用signal/slot的討論,由于回復太多,這里就不貼出原文了。

                    主要是關于怎樣從一個線程發送信號到另外一個線程的問題。其實這個也不是什么復雜的問題,在qt的asstant里面已經描訴的比較清楚了。當我們鏈接信號的時候使用qt::queuedConnection就能使slot在它自己的線程里面運行。

                    另我驚訝的是在其中一個的回復種他給出了一些資料,其中一個名為you‘ar doing it wrong。帖子是英文的,由于英文水平有限,加上他所說的使用QT thread的方式和我們平時直接派生QThread實現run函數的方式不一樣,所以讓我看的非常含糊,甚至到了不清不楚的地步。看了后面的大量的回復和討論,勉強明白了它的意思。

                具體請看這里
            http://labs.qt.nokia.com/2010/06/17/youre-doing-it-wrong/

                   在那里他提出了一種新的使用QThread的方式,其實也不算是信了,據說qt 4.4就已經有了。那就是QObject::moveToThread。根據QT的asstant的描述,moveToThread的作用是把一個QOject移動到一個線程里面去,那么它到底是什么意思呢。我的理解就是當我們調用QObject的moveToThread方法之后,我們這個派生自QObject的類的代碼就會在新的線程里面執行。而那篇文章所說的就是大多數對這個函數產生了誤解,人們總是在派生的QThread的類的構造函數里面調用moveToThread(this)以希望把該類的所有函數都在該線程里面執行。這樣是錯誤的。

                     今天為了驗證這個方法到底有什么用,寫了一些代碼來做測試。

            1、

               

            1. #include <QObject>  
            2. #include <QDebug>  
            3. #include <QThread>  
            4.   
            5. class MyObject : public QObject {  
            6.     Q_OBJECT  
            7.     public:  
            8.         MyObject() {};  
            9.         ~MyObject() {}  
            10.   
            11.     public slots:  
            12.         void first() {  
            13.             qDebug() << QThread::currentThreadId();  
            14.         }  
            15.         void second() {  
            16.             qDebug() << QThread::currentThreadId();  
            17.         }  
            18.         void three() {  
            19.             qDebug() << QThread::currentThreadId();  
            20.         }  
            21. };  

            2、mainwindow.cxx

            1. #include "mainwindow.h"  
            2. #include <QPushButton>  
            3. #include <QVBoxLayout>  
            4. #include "myobject.h"  
            5.   
            6. MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent) {  
            7.     my = new MyObject;  
            8.     firstButton = new QPushButton(tr("first"), 0);  
            9.     connect(firstButton, SIGNAL(clicked()), my, SLOT(first()), Qt::QueuedConnection);  
            10.     secondButton = new QPushButton(tr("second"), 0);  
            11.     connect(secondButton, SIGNAL(clicked()), my, SLOT(second()), Qt::QueuedConnection);  
            12.     threeButton = new QPushButton(tr("three"), 0);  
            13.     connect(threeButton, SIGNAL(clicked()), my, SLOT(three()), Qt::QueuedConnection);  
            14.     selfButton = new QPushButton(tr("self"), 0);  
            15.     connect(selfButton, SIGNAL(clicked()), this, SLOT(onSelfPushed()));  
            16.     exitButton = new QPushButton(tr("exit"), 0);  
            17.     connect(exitButton, SIGNAL(clicked()), this, SLOT(onExitPushed()));  
            18.   
            19.     QVBoxLayout *layout = new QVBoxLayout;  
            20.     layout->addWidget(firstButton);  
            21.     layout->addWidget(secondButton);  
            22.     layout->addWidget(threeButton);  
            23.     layout->addWidget(selfButton);  
            24.     layout->addWidget(exitButton);  
            25.   
            26.     QWidget *p = new QWidget;  
            27.     p->setLayout(layout);  
            28.   
            29.     QThread *thread = new QThread;  
            30.     my->moveToThread(thread);  
            31.   
            32.     thread->start();  
            33.     connect(thread, SIGNAL(started()), my, SLOT(first()));  
            34.   
            35.     setCentralWidget(p);  
            36. }  
            37.   
            38. MainWindow::~MainWindow() {  
            39. }  
            40.   
            41. void MainWindow::onFirstPushed() {  
            42.     my->first();  
            43. }  
            44.   
            45. void MainWindow::onSecondPushed() {  
            46.     my->second();  
            47. }  
            48.   
            49. void MainWindow::onThreePushed() {  
            50.     my->three();  
            51. }  
            52.   
            53. void MainWindow::onSelfPushed() {  
            54.     qDebug() << QThread::currentThreadId();  
            55. }  
            56.   
            57. void MainWindow::onExitPushed() {  
            58.     close();  
            59. }  

                通過測試,在mainwidow.cxx使用上面的代碼的時候,由于my調用了movetothread函數,那么它所有的槽函數都是執行在新開辟的線程里面。

               如果去掉moveToThread函數,那么所有的函數都將執行在gui線程里面。

               同時為了測試connect的第五個參數,在connect的時候可以將Qt::QueuedConnection修改為Qt::DirectConnection,這樣所有的槽函數也將在主線程里面執行。

                 最后要注意的是,如果上面connect的時候連接的是this的onXXXXXX槽函數再來調用的my的槽函數的話,那么這些槽函數也將執行在onXXXXX槽函數所在的線程,這里是主線程。

                 通過上面的測試,我們在使用線程的時候,就可以將一個類派生自QObject,然后實現所有的signal/slot,然后通過調用movetothread函數來使他們執行在新的線程里面,而不是每次都要重新派生QThread,并且派生QThread函數的另外一個不好的地方是只有run函數內部的代碼才會執行在新線程里面,相比起來,派生QObject并使用movetothread函數更具有靈活性。

                最后,把討論中列出的所有的網址列出來哈。

            posted on 2013-12-11 17:08 米米 閱讀(887) 評論(0)  編輯 收藏 引用 所屬分類: qt
            无码人妻久久一区二区三区蜜桃| 国产成人无码精品久久久久免费| 久久久久久噜噜精品免费直播| 伊人久久免费视频| 久久久久久亚洲精品不卡| 久久影视国产亚洲| 欧美精品一区二区精品久久| yy6080久久| 久久久久久综合一区中文字幕| 无码人妻久久一区二区三区蜜桃| 久久久久久国产精品免费无码| 国产毛片久久久久久国产毛片| 2021国内久久精品| 狠狠色伊人久久精品综合网| 国产成人精品久久免费动漫| 久久亚洲精精品中文字幕| 久久无码国产专区精品| 91久久精品电影| 国内精品久久久久久99| 久久久噜噜噜久久熟女AA片| 久久精品女人天堂AV麻| 亚洲国产精品久久久久婷婷老年| 久久精品国产亚洲AV香蕉| 久久综合亚洲鲁鲁五月天| 久久国产成人精品国产成人亚洲| 国产精品女同久久久久电影院| 狠狠色噜噜色狠狠狠综合久久| 国产成人无码精品久久久性色 | 亚洲综合伊人久久综合| 久久93精品国产91久久综合| 色综合久久88色综合天天| 久久精品国产亚洲av麻豆色欲| 亚洲中文久久精品无码ww16| 久久久久亚洲av综合波多野结衣| 久久免费视频一区| 色老头网站久久网| 人妻无码中文久久久久专区| 亚洲精品乱码久久久久久久久久久久 | 三级三级久久三级久久| 久久天天躁狠狠躁夜夜不卡| 久久久国产打桩机|