成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

詳解QT 信號(hào)機(jī)制 (下篇)

移動(dòng)開(kāi)發(fā)
信號(hào)不是Unix中進(jìn)程間通信的信號(hào)。這里的信號(hào)更多地與圖形界面的輸入輸出聯(lián)系在一起(當(dāng)然也可以是不可見(jiàn)的操作)。先來(lái)看內(nèi)容。

繼續(xù) 詳解QT 信號(hào)機(jī)制 (上篇) 的內(nèi)容接續(xù)介紹,本節(jié)介紹的是詳解QT 信號(hào)機(jī)制 (下篇),以下是QMetaObject的定義(為了瀏覽方便,刪除了一部分次要代碼):

  1. class Q_EXPORT QMetaObject  
  2. {  
  3. public:  
  4. QMetaObject( const char * const class_name, QMetaObject *superclass,  
  5. const QMetaData * const slot_data, int n_slots,  
  6. const QMetaData * const signal_data, int n_signals);  
  7. virtual ~QMetaObject();  
  8. int numSlots( bool super = FALSE ) const; /* 反應(yīng)槽的數(shù)量 */  
  9. int numSignals( bool super = FALSE ) const; /* 信號(hào)的數(shù)量 */  
  10. int findSlot( const char *, bool super = FALSE ) const;  
  11. /* 根據(jù)反應(yīng)槽的名稱(chēng)找到其在列表中的索引 */  
  12. int findSignal( const char *, bool super = FALSE ) const;  
  13. /* 根據(jù)信號(hào)的名稱(chēng)找到其在列表中的索引 */  
  14. const QMetaData *slot( int index, bool super = FALSE ) const;  
  15. /* 根據(jù)索引取得反應(yīng)槽的數(shù)據(jù) */  
  16. const QMetaData *signal( int index, bool super = FALSE ) const;  
  17. /* 根據(jù)索引取得信號(hào)的數(shù)據(jù) */  
  18. QStrList slotNames( bool super = FALSE ) const;  
  19. /* 取得反應(yīng)槽列表 */  
  20. QStrList signalNames( bool super = FALSE ) const;  
  21. /* 取得信號(hào)列表 */  
  22. int slotOffset() const;  
  23. int signalOffset() const;  
  24. static QMetaObject *metaObject( const char *class_name );  
  25. private:  
  26. QMemberDict *init( const QMetaData *, int );  
  27. const QMetaData *slotData; /* 反應(yīng)槽數(shù)據(jù)指針 */  
  28. QMemberDict *slotDict; /* 反應(yīng)槽數(shù)據(jù)字典指針 */  
  29. const QMetaData *signalData; /* 信號(hào)數(shù)據(jù)指針*/  
  30. QMemberDict *signalDict; /* 信號(hào)數(shù)據(jù)字典指針*/  
  31. int signaloffset;  
  32. int slotoffset;  
  33. }; 

再看一下QObject中connect的實(shí)現(xiàn)。剝?nèi)ゴ种Γ瘮?shù)中便露出一個(gè)更細(xì)化的函數(shù):connectInternal,它又做了哪些工作呢?讓我們看一下:

  1. void QObject::connectInternal( const QObject *sender, int signal_index,  
  2. const QObject *receiver,  
  3. int membcode, int member_index )  
  4. {  
  5. QObject *s = (QObject*)sender;  
  6. QObject *r = (QObject*)receiver;  
  7. if ( !s->connections ) {  
  8. /* 如果某個(gè)對(duì)象有信號(hào)或反應(yīng)槽但沒(méi)有建立相互連接是不會(huì)建立連接列表的,這樣可減少一些無(wú)謂的資源消耗 */  
  9. s->connections = new QSignalVec( 7 );  
  10. s->connections->setAutoDelete( TRUE );  
  11. /* 無(wú)連接時(shí),連接列表將被自動(dòng)刪除 */  
  12. }  
  13. QConnectionList *clist = s->connections->at( signal_index );  
  14. if ( !clist ) {  
  15. /* 建立與信號(hào)源對(duì)象中某一個(gè)信號(hào)所對(duì)應(yīng)的接收對(duì)象的列表 */  
  16. clist = new QConnectionList;  
  17. clist->setAutoDelete( TRUE );  
  18. s->connections->insert( signal_index, clist );  
  19. }  
  20. QMetaObject *rrmeta = r->metaObject();  
  21. switch ( membcode ) {  
  22. /* 取得信號(hào)或反應(yīng)槽的數(shù)據(jù)指針 */  
  23. case QSLOT_CODE:  
  24. rm = rmeta->slot( member_index, TRUE );  
  25. break;  
  26. case QSIGNAL_CODE:  
  27. rm = rmeta->signal( member_index, TRUE );  
  28. break;  
  29. }  
  30. QConnection *c = new QConnection( r, member_index,  
  31. rm ? rm->name : "qt_invoke", membcode );  
  32. /* 創(chuàng)建一個(gè)新的信號(hào)/反應(yīng)槽連接 */  
  33. clist->append( c ); /* 信號(hào)源端加入這一對(duì)連接 */  
  34. if ( !r->senderObjects ) {  
  35. /* 類(lèi)似于信號(hào)源端,反應(yīng)槽端的連接列表也是動(dòng)態(tài)創(chuàng)建的 */  
  36. r->senderObjects = new QObjectList;  
  37. }  
  38. r->senderObjects->append( s ); /* 反應(yīng)槽端加入這一對(duì)連接 */  

到此,信號(hào)與反應(yīng)槽的連接已建立完畢,那么信號(hào)產(chǎn)生時(shí)又是如何觸發(fā)反應(yīng)槽的呢?從QObject的定義中可以看出其有多個(gè)activate_signal的成員函數(shù),這些函數(shù)都是protected的,也即只有其自身或子類(lèi)才可以使用。看一下它的實(shí)現(xiàn):

  1. void QObject::activate_signal( QConnectionList *clist, QUObject *o )  
  2. {  
  3. if ( !clist ) /* 有效性檢查 */  
  4. return;  
  5. QObject *object;  
  6. QConnection *c;  
  7. if ( clist->count() == 1 ) {  
  8. /* 對(duì)某一個(gè)對(duì)象的一個(gè)具體信號(hào)來(lái)說(shuō),一般只有一種反應(yīng)槽與之相連,這樣事先判斷一下可以加快處理速度 */  
  9. c = clist->first();  
  10. object = c->object();  
  11. sigSender = this;  
  12. if ( c->memberType() == QSIGNAL_CODE )  
  13. object->qt_emit( c->member(), o ); /* 信號(hào)級(jí)連 */  
  14. else  
  15. object->qt_invoke( c->member(), o );/* 調(diào)用反應(yīng)槽函數(shù) */  
  16. } else {  
  17. QConnectionListIt it(*clist);  
  18. while ( (c=it.current()) ) { /* 有多個(gè)連接時(shí),逐一掃描 */  
  19. ++it;  
  20. object = c->object();  
  21. sigSender = this;  
  22. if ( c->memberType() == QSIGNAL_CODE )  
  23. object->qt_emit( c->member(), o ); /* 信號(hào)級(jí)連 */  
  24. else  
  25. object->qt_invoke( c->member(), o ); /* 調(diào)用反應(yīng)槽函數(shù) */  
  26. }  
  27. }  

至此我們已經(jīng)可以基本了解Qt中信號(hào)/反應(yīng)槽的流程。我們?cè)倏匆幌翾t為此而新增的語(yǔ)法:三個(gè)關(guān)鍵字:slots、signals和emit,三個(gè)宏:SLOT()、SIGNAL()和Q_OBJECT。在頭文件qobjectdefs.h中,我們可以看到這些新增語(yǔ)法的定義如下:

  1. #define slots // slots: in class  
  2. #define signals protected // signals: in class  
  3. #define emit // emit signal  
  4. #define SLOT(a) "1"#a  
  5. #define SIGNAL(a) "2"#a 

由此可知其實(shí)三個(gè)關(guān)鍵字沒(méi)有做什么事情,而SLOT()和SIGNAL()宏也只是在字符串前面簡(jiǎn)單地加上單個(gè)字符,以便程序僅從名稱(chēng)就可以分辨誰(shuí)是信號(hào)、誰(shuí)是反應(yīng)槽。中間編譯程序moc.exe則可以根據(jù)這些關(guān)鍵字和宏對(duì)相應(yīng)的函數(shù)進(jìn)行“翻譯”,以便在C++編譯器中編譯。剩下一個(gè)宏Q_OBJECT比較復(fù)雜,它的定義如下:

  1. #define Q_OBJECT \  
  2. publi \  
  3. virtual QMetaObject *metaObject() const { \  
  4. return staticMetaObject(); \  
  5. }  
  6. \  
  7. virtual const char *className() const; \  
  8. virtual void* qt_cast( const char* ); \  
  9. virtual bool qt_invoke( int, QUObject* ); \  
  10. virtual bool qt_emit( int, QUObject* ); \  
  11. QT_PROP_FUNCTIONS  
  12. \  
  13. static QMetaObject* staticMetaObject(); \  
  14. QObject* qObject() { return (QObject*)this; } \  
  15. QT_TR_FUNCTIONS  
  16. \  
  17. private: \  
  18. static QMetaObject *metaObj; 

從定義中可以看出該宏的作用有兩個(gè):一是對(duì)與自己相關(guān)的QMetaObject中間類(lèi)操作進(jìn)行聲明,另一個(gè)是對(duì)信號(hào)的釋放操作和反應(yīng)槽的激活操作進(jìn)行聲明。當(dāng)moc.exe對(duì)頭文件進(jìn)行預(yù)編譯之后,將會(huì)產(chǎn)生一個(gè)可供C++編譯器編譯的源文件。以上述的Demo類(lèi)為例,假設(shè)它的代碼文件分別為d e m o . h和d e m o . c p p ,預(yù)編譯后將產(chǎn)生

moc_demo.cpp,其主要內(nèi)容如下:

  1. QMetaObject *Demo::metaObj = 0;  
  2. void Demo::initMetaObject()  
  3. {  
  4. if ( metaObj )  
  5. return;  
  6. if ( strcmp(QObject::className(), "QObject") != 0 )  
  7. badSuperclassWarning("Demo","QObject");  
  8. (void) staticMetaObject();  
  9. }  
  10. QMetaObject* Demo::staticMetaObject()  
  11. {  
  12. if ( metaObj )  
  13. return metaObj;  
  14. (void) QObject::staticMetaObject();  
  15. typedef void(Demo::*m1_t0)(int);  
  16. m1_t0 v1_0 = Q_AMPERSAND Demo::setValue; /* 定位反應(yīng)槽的入口 */  
  17. QMetaData *slot_tbl = QMetaObject::new_metadata(1);  
  18. /* 新建一個(gè)反應(yīng)槽數(shù)據(jù) */  
  19. QMetaData::Access *slot_tbl_access = QMetaObject::new_metaaccess(1);  
  20. slot_tbl[0].name = "setValue(int)"; /* 反應(yīng)槽名稱(chēng) */  
  21. slot_tbl[0].ptr = *((QMember*)&v1_0);  
  22. /* 通過(guò)反應(yīng)槽名稱(chēng)可以找到反應(yīng)槽的入口指針 */  
  23. slot_tbl_access[0] = QMetaData::Public; /* 權(quán)限類(lèi)型 */  
  24. typedef void(Demo::*m2_t0)(int);  
  25. m2_t0 v2_0 = Q_AMPERSAND Demo::valueChanged; /* 定位信號(hào)的入口 */  
  26. QMetaData *signal_tbl = QMetaObject::new_metadata(1); /* 新建信號(hào)數(shù)據(jù) */  
  27. signal_tbl[0].name = "valueChanged(int)"; /* 信號(hào)名稱(chēng) */  
  28. signal_tbl[0].ptr = *((QMember*)&v2_0);  
  29. /* 通過(guò)信號(hào)名稱(chēng)可以找到信號(hào)的入口指針 */  
  30. metaObj = QMetaObject::new_metaobject(  
  31. /* 創(chuàng)建一個(gè)與demo類(lèi)相關(guān)的QMetaObject對(duì)象 */  
  32. "Demo", "QObject",  
  33. slot_tbl, 1,  
  34. signal_tbl, 1,  
  35. 0, 0 );  
  36. metaObj->set_slot_access( slot_tbl_access ); /* 設(shè)置權(quán)限 */  
  37. return metaObj;  
  38. }  
  39. // 有信號(hào)時(shí)即激活對(duì)應(yīng)的反應(yīng)槽或另一個(gè)信號(hào)  
  40. void Demo::valueChanged( int t0 )  
  41. {  
  42. activate_signal( "valueChanged(int)", t0 );  

該文件中既沒(méi)有Qt特有的關(guān)鍵字,也沒(méi)有特殊的宏定義,完全符合普通的C++語(yǔ)法,因此可以順利編譯和鏈接。

小結(jié):關(guān)于詳解QT 信號(hào)機(jī)制 (下篇)的內(nèi)容介紹完了,希望本文對(duì)你有所幫助!

責(zé)任編輯:zhaolei 來(lái)源: 互聯(lián)網(wǎng)
相關(guān)推薦

2011-07-05 18:32:52

QT 信號(hào) 機(jī)制

2024-07-16 10:52:09

2025-02-03 07:00:00

2021-12-10 00:01:53

Vsync信號(hào)機(jī)制

2011-06-17 10:19:11

Qt QWidge QSetting

2011-06-17 09:58:26

Qt Chapter QObject

2011-07-01 14:20:59

Qt 事件

2011-06-23 13:38:27

QT 元對(duì)象 信號(hào)

2011-06-09 09:45:35

Linux QT 信號(hào)

2011-06-30 10:36:07

QT Graphics View

2011-06-23 14:40:13

Qt 信號(hào)

2011-06-23 14:05:32

Qt 事件機(jī)制

2022-11-03 07:35:47

OS內(nèi)核異步

2017-09-14 09:40:32

PythonUbuntu信號(hào)機(jī)制

2011-06-24 16:50:06

QT 框架 OpenCV

2011-04-07 17:43:37

Shapping

2011-04-07 17:54:22

Policing

2011-06-29 17:20:20

Qt 內(nèi)存 QOBJECT

2011-06-13 10:21:25

QT 信號(hào) 槽機(jī)制

2011-06-23 18:16:59

Qt Qtopia
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 亚洲欧美成人影院 | 欧美成视频 | h小视频| 欧美国产日韩在线 | 成人a视频| 久久69精品久久久久久久电影好 | 亚洲精品乱码久久久久久9色 | xx性欧美肥妇精品久久久久久 | 日韩欧美二区 | 久久91av| 国产亚洲黄色片 | 国产精品性做久久久久久 | 欧美日韩在线一区 | 国产一区不卡在线观看 | 欧美区在线观看 | 911网站大全在线观看 | 亚洲福利一区二区 | 久久国产精品久久国产精品 | 国产精品成人久久久久 | 噜久寡妇噜噜久久寡妇 | 久久一级| 夜夜草| 亚州影院| 久久伊人精品 | 亚洲综合一区二区三区 | 免费看日韩视频 | 久久成人一区 | 女同久久另类99精品国产 | 狠狠躁夜夜躁人人爽天天高潮 | 成人av播放 | 一区二区三区视频在线 | 一区中文字幕 | chinese中国真实乱对白 | 一区二区精品 | 亚洲精品视频网站在线观看 | 日韩一二区在线观看 | 久久69精品久久久久久久电影好 | 欧美白人做受xxxx视频 | 久久www免费人成看片高清 | 国产精品久久一区 | 国产精品永久久久久久久www |