• <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>
            隨筆-1  評(píng)論-10  文章-0  trackbacks-0

            第一回 Signal和Slot是同步的還是異步的?

              我們知道Qt以他的signal和slot機(jī)制獨(dú)步天下。但大家在用的時(shí)候有沒有注意過,signal和slot之間是異步的,還是同步的呢?為此我問過不少使用Qt的道友。有人說是同步的,有人說是異步的,也有人說要看當(dāng)時(shí)你的人品。:( #$%^&*

              為此貧道,特別做了以下幾個(gè)測(cè)試:

                 First,在main()主函數(shù)里,設(shè)置兩個(gè)基于QObject為父類的對(duì)象a和b,a觸發(fā)signal,b接受signal。請(qǐng)看具體案例:

                

             1 class MyTestA : public QObject
            2 {
            3 Q_OBJECT
            4 public:
            5 void emitSignal()
            6 {
            7 signalMyTestA();
            8 }
            9
            10 public slots:
            11 void slotMyTestA()
            12 {
            13 qDebug()<<"slotMyTestA is called.";
            14 }
            15 signals:
            16 void signalMyTestA();
            17 };
            18
            19 class MyTestB : public QObject
            20 {
            21 Q_OBJECT
            22 public slots:
            23 void slotMyTestB()
            24 {
            25 qDebug()<<"slotMyTestB is called.";
            26 }
            27 signals:
            28 void signalMyTestB();
            29 };
            30
            31 int main(int argc, char *argv[])
            32 {
            33 QApplication app(argc, argv);
            34
            35 MyTestA a;
            36 MyTestB b;
            37 QObject::connect(&a,SIGNAL(signalMyTestA()),&b,SLOT(slotMyTestB()));
            38
            39 a.emitSignal();
            40
            41 return app.exec();
            42 }

            在slotMyTestB的函數(shù)里打個(gè)斷點(diǎn),看一下調(diào)用堆棧(call stack)。

            是同步調(diào)用的,某些道友開始拈胡微笑,實(shí)踐出真知啊。

            此時(shí)只見東方黑云滾滾,電閃雷鳴,又有道友開始度劫了。突然一度劫道友橫眉冷對(duì),拿起拂塵刷刷的改寫了上面的代碼。只見此道友把a(bǔ)對(duì)象挪到了一個(gè)新線程中(MyTestC創(chuàng)建的),而b對(duì)象仍然在主線程中。然后a對(duì)象觸發(fā)信號(hào)。

            class MyTestA : public QObject
            {
                Q_OBJECT
            public:
                void emitSignal()
                {
                    signalMyTestA();
                }
            public slots:
                void slotMyTestA()
                {
                    qDebug()<<"slotMyTestA is called.";
                }
            signals:
                void signalMyTestA();
            };
            class MyTestB : public QObject
            {
                Q_OBJECT
            public slots:
                void slotMyTestB()
                {
                    qDebug()<<"slotMyTestB is called.";
                }
            signals:
                void signalMyTestB();
            };
            extern MyTestB *g_pMyTestB;
            class MyTestC : public QThread
            {
                Q_OBJECT
            public:
                void run()
                {
                    MyTestA a;
                    connect(&a,SIGNAL(signalMyTestA()),g_pMyTestB,SLOT(slotMyTestB()));
                    a.emitSignal();
                    exec();
                }
            public slots:
                void slotMyTestC()
                {
                    qDebug()<<"slotMyTestC is called.";
                }
            signals:
                void signalMyTestC();
            };
            class MyTest : public QDialog
            {
                Q_OBJECT
            public:
                MyTest(QWidget *parent = 0, Qt::WFlags flags = 0);
                ~MyTest();
            private:
                Ui::MyTestClass ui;
            };
            ////////////////////////////////////////////////
            MyTestB *g_pMyTestB = NULL;
            int main(int argc, char *argv[])
            {
                QApplication app(argc, argv);
                MyTestB b;
                g_pMyTestB = &b;
                MyTestC c;
                c.start();
                return app.exec();
            }
            

            說時(shí)遲,那時(shí)快。在一道紫雷劈下之際,按下了F5。只見,此時(shí)的調(diào)用堆棧顯示,

            奇跡出現(xiàn)了,居然變成異步調(diào)用了。只見東方天空萬道金光射下,在陣陣仙樂聲中,傳來朗朗之聲:"貧道塵事已了,再無牽掛"。

            難道Qt真的是靠人品的,或者Qt莫不是也是修仙道友,不日也將飛升。

            在吾等眾人膜拜加疑惑之時(shí),只見飛升前輩,留下一條偈語。內(nèi)事不決問百度,外事不決問谷歌。

            吾等眾人立刻搜尋,恍然大物。

            原來signal和slot是異步調(diào)用還是同步調(diào)用,取決于對(duì)connect的設(shè)定。其實(shí)connect還有一個(gè)參數(shù)(Qt::ConnectionType),是它決定了是同步還是異步。以下是ConnectionType的定義

            只不過,平常它有一個(gè)默認(rèn)值Qt::AutoConnection,我們忽略了它。這時(shí)有道友問道,為何在AutoConnection模式下,有時(shí)是同步,有時(shí)是異步,莫非Auto就是人品代名詞。

            非也,其實(shí)Auto是這樣規(guī)定的,

            當(dāng)sender和receiver在同一線程時(shí),就是同步模式,而在不同線程時(shí),則是異步模式。

            眾人皆曰善。

            就在眾人彈冠相慶之時(shí),突然一道類似眼鏡發(fā)出的寒光閃過,一個(gè)黑影漸漸清晰了起來。

            他居然就是..................

            青春永駐,十二年如一日的柯南君,他招牌式的磁性聲音給眾道友一晴天霹靂,“諸位以為這就是全部的真相嗎?”

            接著他刷刷的又改寫了代碼,在主線程中生成a,b兩個(gè)對(duì)象,而a對(duì)象在新線程(MyTestC創(chuàng)建的)中觸發(fā)信號(hào)。

             

            class MyTestA : public QObject
            {
                Q_OBJECT
            public:
                void emitSignal()
                {
                    signalMyTestA();
                }
            public slots:
                void slotMyTestA()
                {
                    qDebug()<<"slotMyTestA is called.";
                }
            signals:
                void signalMyTestA();
            };
            class MyTestB : public QObject
            {
                Q_OBJECT
            public slots:
                void slotMyTestB()
                {
                    qDebug()<<"slotMyTestB is called.";
                }
            signals:
                void signalMyTestB();
            };
            extern MyTestB *g_pMyTestB;
            extern MyTestA *g_pMyTestA;
            class MyTestC : public QThread
            {
                Q_OBJECT
            public:
                void run()
                {
                    g_pMyTestA->emitSignal();
                    exec();
                }
            public slots:
                void slotMyTestC()
                {
                    qDebug()<<"slotMyTestC is called.";
                }
            signals:
                void signalMyTestC();
            };
            /////////////////////////////////////////////
            MyTestB *g_pMyTestB = NULL;
            MyTestA *g_pMyTestA = NULL;
            int main(int argc, char *argv[])
            {
                QApplication app(argc, argv);
                MyTestA a;
                g_pMyTestA = &a;
                MyTestB b;
                g_pMyTestB = &b;
                QObject::connect(&a,SIGNAL(signalMyTestA()),&b,SLOT(slotMyTestB()));
                MyTestC c;
                c.start();
                return app.exec();
            }
            

             

            在眾人疑惑的眼光中,此君淡定的按下了F5。只見調(diào)用堆棧(call stack)顯示

            眾人皆驚呼,“Impossible”。a和b明明是屬于一個(gè)線程的,為何會(huì)異步調(diào)用。此時(shí)我們熟悉的語錄,又在耳邊回響,是"我相信真相只有一個(gè)!!!"這句話嗎?No,只見柯南君,優(yōu)雅地?fù)]了揮手指,"Nothing impossible",從口中緩緩滑出。

            。眾人皆撲街,“有屁快放”。

            此時(shí)柯南君緩緩從口袋中,摸出一張紙,拋向空中,然后轉(zhuǎn)身離去。只見隨風(fēng)飄落的紙面上面摘錄了這么一段Qt源代碼,在Auto模式下,如果要同步調(diào)用,不僅要求sender和receiver是同一線程,而且sender觸發(fā)的時(shí)候,所在的線程也要和receiver一致。

             // determine if this connection should be sent immediately or
                        // put into the event queue
                        if ((c->connectionType == Qt::AutoConnection
                             && (currentThreadData != sender->d_func()->threadData
                                 || receiver->d_func()->threadData != sender->d_func()->threadData))
                            || (c->connectionType == Qt::QueuedConnection)) {
                            queued_activate(sender, signal_absolute_index, c, argv ? argv : empty_argv);
                            continue;
                        } else if (c->connectionType == Qt::BlockingQueuedConnection) {
                            blocking_activate(sender, signal_absolute_index, c, argv ? argv : empty_argv);
                            continue;
                        }

            摘自qobject.cpp

            z眾人皆驚,原來在Auto模式下,如果sender的觸發(fā)時(shí)所處的線程和receiver不同,也會(huì)是異步調(diào)用。此時(shí)道友齊聲向柯南喊道“這是全部的真相了嗎”?柯南轉(zhuǎn)過頭來笑而不語,漸漸又消失在黑暗中。“有多少無恥可以重來”漂上了眾人的心頭。望著遠(yuǎn)處的雨后陽光,一個(gè)大大的問號(hào)也出現(xiàn)在眾人頭頂,"Qt你到底有多無恥???"。眾人又陷入了沉思。

            欲知后事如何,請(qǐng)聽下回分解。

             

            此篇已在CNBLOG同時(shí)發(fā)布

             

             

             

             

             

             




            posted on 2011-08-26 10:41 櫻桃小錘子 閱讀(12945) 評(píng)論(10)  編輯 收藏 引用

            評(píng)論:
            # re: Qt那點(diǎn)事兒(一) 2011-08-26 12:25 | 飯中淹
            磁性的柯南。。。看著他變小的長(zhǎng)大的我們這一代。  回復(fù)  更多評(píng)論
              
            # re: Qt那點(diǎn)事兒(一) 2011-08-26 13:35 | tinyms
            Sign是異步的,slot是同步的,前者是消息,后者僅僅是回調(diào)  回復(fù)  更多評(píng)論
              
            # re: Qt那點(diǎn)事兒(一) 2011-08-26 14:27 | 櫻桃小錘子
            @tinyms
            Sign是異步的,slot是同步的>>何解,請(qǐng)賜教
              回復(fù)  更多評(píng)論
              
            # re: Qt那點(diǎn)事兒(一) 2011-08-26 15:23 | 他她女鞋
            感覺還是挺繁雜的。  回復(fù)  更多評(píng)論
              
            # re: Qt那點(diǎn)事兒(一)[未登錄] 2011-08-26 17:20 | cc
            我弟,你敢說的簡(jiǎn)單點(diǎn)嗎  回復(fù)  更多評(píng)論
              
            # re: Qt那點(diǎn)事兒(一) 2011-08-28 14:12 | mos
            搞這么復(fù)雜,真不如boost::signal  回復(fù)  更多評(píng)論
              
            # re: Qt那點(diǎn)事兒(一) 2011-08-29 09:45 | zuhd
            哥 你的文筆太有愛了 不懂qt
            我是來看小說的  回復(fù)  更多評(píng)論
              
            # re: Qt那點(diǎn)事兒(一)[未登錄] 2011-08-29 11:19 | 姚冬
            如果比較下常用的幾種 signal/slot實(shí)現(xiàn)的話,我覺得Qt的實(shí)現(xiàn)是最好的。

            boost的signal/slot 有一個(gè)很嚴(yán)重的問題,就是會(huì)導(dǎo)致編譯非常慢,你寫個(gè)小測(cè)試程序是不會(huì)感覺到的,如果在幾十個(gè)文件中使用的話,編譯時(shí)間會(huì)成倍增長(zhǎng),即使用并行編譯也是慢。都是模板搞的,boost的泛型用得太花哨了。

            Qt則完全沒有這個(gè)問題,而且執(zhí)行效率也還可以。  回復(fù)  更多評(píng)論
              
            # re: Qt那點(diǎn)事兒(一) 2011-08-30 13:52 | uestc001@163.com
            文筆不錯(cuò)  回復(fù)  更多評(píng)論
              
            # re: Qt那點(diǎn)事兒(一) 2011-11-07 21:54 | 渡劫導(dǎo)游
            某學(xué)過半個(gè)月Qt的導(dǎo)游落果  回復(fù)  更多評(píng)論
              

            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            国产91色综合久久免费分享| 伊人热热久久原色播放www| 青青草国产成人久久91网| 欧美久久亚洲精品| 久久国产色AV免费观看| 欧美粉嫩小泬久久久久久久| 久久久久亚洲AV无码永不| 久久久中文字幕日本| 久久99国内精品自在现线| 久久夜色精品国产亚洲av| 久久精品国产影库免费看| 性高湖久久久久久久久| 久久久久这里只有精品| 久久精品免费观看| 久久综合给合久久狠狠狠97色| 久久高清一级毛片| 久久精品视频免费| 久久国产色AV免费观看| 一本色道久久综合亚洲精品| 青青久久精品国产免费看| 国内精品久久久久久不卡影院 | 国产精品激情综合久久| 久久九九精品99国产精品| 久久婷婷国产剧情内射白浆| 久久国产成人午夜aⅴ影院| 精品久久久久久综合日本| 久久久久亚洲av无码专区导航| 四虎国产精品成人免费久久| 久久国产精品国语对白| 伊人久久大香线焦综合四虎 | 国产免费久久精品99re丫y| 久久精品国产亚洲一区二区三区| 久久精品国产只有精品2020| 精品国产一区二区三区久久久狼| 中文字幕日本人妻久久久免费| 波多野结衣久久一区二区| 亚洲国产视频久久| 国产美女亚洲精品久久久综合| 久久天天躁狠狠躁夜夜不卡| 99久久免费国产精品特黄| 一本久久a久久精品亚洲|