• <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
            色欲久久久天天天综合网精品| 精品久久久久久亚洲精品| 伊人久久综在合线亚洲2019| 办公室久久精品| 久久精品中文无码资源站| av国内精品久久久久影院| 日韩久久久久中文字幕人妻 | 亚洲欧美一级久久精品| 国产精品久久久久久久app | 久久亚洲AV成人无码软件| 国产精品久久久久久福利69堂| 久久婷婷五月综合成人D啪| 亚洲国产精品18久久久久久| 精品久久久久久无码中文字幕 | 99久久精品这里只有精品| 久久天天躁夜夜躁狠狠躁2022| 韩国三级大全久久网站| 区久久AAA片69亚洲| 色婷婷狠狠久久综合五月| 欧美日韩中文字幕久久伊人| 久久久久亚洲av成人网人人软件| 国产激情久久久久影院小草 | 久久久久九九精品影院| avtt天堂网久久精品| 久久综合九色综合网站| 日本久久中文字幕| 久久噜噜久久久精品66| 久久免费99精品国产自在现线| 婷婷综合久久狠狠色99h| AV狠狠色丁香婷婷综合久久 | 精品久久久无码中文字幕天天| 久久亚洲精品无码AV红樱桃| 亚洲国产精品久久电影欧美| 97久久国产露脸精品国产| 国产一区二区久久久| 久久乐国产综合亚洲精品| 久久亚洲高清综合| 亚洲国产成人精品无码久久久久久综合| 久久综合九色综合久99| 国产69精品久久久久99| 国产精品成人精品久久久|