從零開(kāi)始學(xué)Qt(四)信號(hào)與槽
信號(hào)與槽
????????????????????????????????????????????????????????????????????????????????????????????????? -----書不記,熟讀可記;義不精,細(xì)思可精。
1、信號(hào)/槽是啥?
??????? 古有“烽火狼煙”傳遞消息,敵人來(lái)犯的消息迅速傳達(dá)開(kāi)來(lái),是多么的聰慧啊。煙就是信號(hào);下一個(gè)燃火臺(tái)看到煙后就點(diǎn)燃燃料,這個(gè)就槽。換種方法說(shuō)就是 一個(gè)按鈕被點(diǎn)擊了,會(huì)觸發(fā)一個(gè)點(diǎn)擊的信號(hào);槽是指收到信號(hào)之后,具體要去做什么。
??????? 專業(yè)點(diǎn)說(shuō)你可以從設(shè)計(jì)模式上理解,其實(shí)就是觀察者模式。信號(hào)是“主題”,每個(gè)槽是“觀察者”。當(dāng)發(fā)出信號(hào)的時(shí)候,對(duì)該信號(hào)感興趣的槽就會(huì)被觸發(fā)(有沒(méi)有覺(jué)得突然逼格變高了)。簡(jiǎn)單點(diǎn)說(shuō)信號(hào)類似廣播,每個(gè)槽都可以聽(tīng)到這個(gè)廣播,但是只有你連接過(guò)的槽才會(huì)去接收這個(gè)信號(hào),并作出相應(yīng)的操作。
????? ? 信號(hào)和槽分為兩種,一種是類自帶的信號(hào)和槽,直接可以使用。另外一種是我們可以自定義信號(hào)和槽,使得我們更加的自由。當(dāng)一個(gè)類被繼承的時(shí),該類的信號(hào)和槽也同時(shí)被繼承。
2、怎么查看Qt中具體類的信號(hào)和槽
??????? 有兩種方法可以查看,第一種我們可以打開(kāi)Qt Assistant(助手),還有一種是直接打開(kāi)Qt Creator(里面也集成了上面那個(gè)模塊),所以我們直接講后者。跟著我的步驟,我們以查看QLabel為例子,如下圖:
????????點(diǎn)擊上圖的信號(hào)/槽,就會(huì)跳到具體的位置。
3、打開(kāi)之前的HelloQt工程
??????? 我們這篇的內(nèi)容會(huì)圍繞著這個(gè)例子來(lái)講《從零開(kāi)始學(xué)Qt(三)從Hello,Qt說(shuō)起》。所以我們來(lái)打開(kāi)之前的工程。并且介紹下,如何打開(kāi)Qt工程。
??????? 打開(kāi)Qt有兩種方法:
1)第一種是打開(kāi)工程目錄,然后雙擊 HelloQt.pro 文件,就可以直接打開(kāi)該工程。
2)先打開(kāi)Qt Creator,從文件->打開(kāi)文件或項(xiàng)目,找到HelloQt工程,然后選中HelloQt.pro文件,然后點(diǎn)擊打開(kāi)。
4、自定義信號(hào)跟槽
??????? 類聲明中必須包含Q_OBJECT宏;信號(hào)跟槽都必須在頭文件中定義。
??????? 1)信號(hào): 只要再.h中聲明即可,發(fā)射信號(hào)用emit關(guān)鍵字。
??????? 2)槽:實(shí)現(xiàn)的時(shí)候跟普通的成員函數(shù)一樣實(shí)現(xiàn)即可。
5、格式
???????? connect(發(fā)送者,信號(hào),接收者,槽);
??????? 1)一個(gè)信號(hào)可以與另外一個(gè)信號(hào)相連
??????? connect(發(fā)送者,信號(hào),接收者,信號(hào));
??????? 2)一個(gè)信號(hào)可以與多個(gè)槽相連
????????connect(發(fā)送者1,信號(hào)1, 接收者1, 槽1);
??????? connect(發(fā)送者1, 信號(hào)1, 接收者2, 槽2);
??????? 3)一個(gè)槽可以被多個(gè)信號(hào)關(guān)聯(lián)。
????????connect(發(fā)送者1,信號(hào)1, 接收者1, 槽1);
??????? connect(發(fā)送者2, 信號(hào)2, 接收者1, 槽1);
6、例子
??????? 1)添加一個(gè)按鈕,如圖下圖所示操作:
???????????
??????? 2)Ctrl + S 保存
??????? 3)點(diǎn)擊回到編輯界面,如下圖:
??????? 3)在.h里面自定義槽函數(shù),響應(yīng)按鈕按下的信號(hào)
??????????????? 雙擊mainwindow.h;在編輯窗口看到mainwindow.h文件。如下圖:
#ifndef?MAINWINDOW_H #define?MAINWINDOW_H #includenamespace?Ui?{ class?MainWindow; } class?MainWindow?:?public?QMainWindow { ????Q_OBJECT public: ????explicit?MainWindow(QWidget?*parent?=?0); ????~MainWindow(); public?slots:???????????????????//所有的槽函數(shù)都必須申明public?slots語(yǔ)句內(nèi) ????void?slt_clearLabel();??????//這里定義一個(gè)槽函數(shù),來(lái)實(shí)現(xiàn)清空文本框的內(nèi)容 private: ????Ui::MainWindow?*ui; }; #endif?//?MAINWINDOW_H
??????? 4)實(shí)現(xiàn)槽函數(shù)
???????????? 上面也提到了,實(shí)現(xiàn)槽函數(shù)。跟實(shí)現(xiàn)普通的成員函數(shù)一樣。
void?MainWindow::slt_clearLabel() { ????ui->label->clear();?????//清空文本框內(nèi)容 }
??????? 5)綁定????
connect(ui->pushButton,?SIGNAL(pressed()),?this,?SLOT(slt_clearLabel()));
??????? 6)mainwindow.cpp文件的完整代碼
#include?"mainwindow.h" #include?"ui_mainwindow.h" MainWindow::MainWindow(QWidget?*parent)?: ????QMainWindow(parent), ????ui(new?Ui::MainWindow) { ????ui->setupUi(this); ????connect(ui->pushButton,?SIGNAL(pressed()),?this,?SLOT(slt_clearLabel())); } MainWindow::~MainWindow() { ????delete?ui; } void?MainWindow::slt_clearLabel() { ????ui->label->clear();?????//清空文本框內(nèi)容 }
??? 7)運(yùn)行結(jié)果
??????? 點(diǎn)擊按鈕的時(shí)候,QLabel里面的文字會(huì)被清空。
7、Qt5中的新語(yǔ)法
??????? 這里先簡(jiǎn)單介紹一部分。介紹是因?yàn)镼t5里面更新的語(yǔ)法,有些比較實(shí)用。為什么先介紹上面的,是因?yàn)榕f版本對(duì)于新手比較好理解。
??????? 1)先來(lái)一個(gè)比較。一個(gè)QSlider對(duì)象的valueChanged信號(hào)鏈接到一個(gè)QSpinBox對(duì)象的setValue槽,使用
傳統(tǒng)方式:
connect(slider,?SIGNAL(valueChanged(int)),?spinbox,?SLOT(setValue(int)));
使用新式語(yǔ)法格式:
connect(slider,?&QSlider::valueChanged,?spinbox,?&QSpinBox::setValue);
??????? 2)新語(yǔ)法的優(yōu)點(diǎn)
編譯期:會(huì)檢查信號(hào)和槽是否存在,且參數(shù)類型是否合法,如果錯(cuò)誤將編譯不通過(guò)。(舊版的如果參數(shù)不匹配只會(huì)有個(gè)Warning的信息,而編譯可以通過(guò)) 信號(hào)可以和槽函數(shù)、普通函數(shù)、類的成員函數(shù)、lambda函數(shù)連接。(舊版的只能跟槽函數(shù)連接)允許一些自動(dòng)類型轉(zhuǎn)換,即信號(hào)和槽參數(shù)類型不必完全匹配。(舊版的一定要匹配,否則連接失敗。槽函數(shù)無(wú)法響應(yīng))。例如:(舊版的會(huì)連接失?。?pre>connect(a,?SIGNAL(sig(int)),?b,?SLOT(slt(double)));參數(shù)可以是typedef,如下:(舊版的會(huì)連接失敗)
typedef?int?myInt; connect(a,?SIGNAL(sig(int)),?b,?SLOT(slt(myInt)));
8、作業(yè)
????? 你們可以試著,把例子里面的舊版連接,改成Qt5新語(yǔ)法的連接。(答案在下篇博客)
結(jié)束:
??????? 最后有同學(xué)會(huì)遇到問(wèn)題,可以加QQ群討論。如果我哪里錯(cuò)了,也希望有人告知我,我來(lái)修改文章,以免誤導(dǎo)他人。