我用Qt寫(xiě)程序也兩年多了局蚀,主要是用這玩意做課程大作業(yè)喷兼、課程設(shè)計(jì)之類的玩意晒屎。期間遇到過(guò)很多奇怪的問(wèn)題,這里做一下整理钉鸯。
2017-01-31 更新
2016-12-18 更新
有毒的頭文件
我在項(xiàng)目里定義了一個(gè)頭文件 global.h吧史,一開(kāi)始工程可以正常地編譯運(yùn)行。但是當(dāng)多個(gè)其他的頭文件開(kāi)始包含global.h的時(shí)候唠雕,奇怪的時(shí)候開(kāi)始出現(xiàn)了贸营,編譯運(yùn)行各種報(bào)錯(cuò),大概意思就是符號(hào)Undifined及塘,特別是global.h文件里的變量怎么也找不到莽使,但是語(yǔ)法高亮又是正常的。折騰一通后笙僚,發(fā)現(xiàn)是global.h的預(yù)處理頭的問(wèn)題:
#ifndef GLOBAL_H
#define GLOBAL_H
/** src **/
#endif
發(fā)現(xiàn)GLOBAL_H這個(gè)宏很可能在Qt內(nèi)部的頭文件中被定義過(guò)了芳肌,所以只要把GLOBAL_H換一個(gè)名字就行了,可是這是Qt自動(dòng)幫我生成的啊肋层,好坑爹亿笤。
神奇的重構(gòu)功能
一旦項(xiàng)目中文件多了之后,想要給某個(gè)變量或方法進(jìn)行重命名栋猖,手動(dòng)操作出錯(cuò)概率太高净薛,免不了要用IDE提供的重構(gòu)(Refactor)功能。Qt這玩意的變量重構(gòu)似乎不是基于語(yǔ)法樹(shù)分析的蒲拉,好像只是在文本層面進(jìn)行批量替換肃拜。
我為什么這么猜測(cè)呢,因?yàn)槲矣龅竭^(guò)一件坑爹事雌团。有一次工程里有一個(gè)變量名比較特殊燃领,大概是像qstring、os锦援、process之類的變量名猛蔽,具體叫什么我不記得了鸵贬,我對(duì)這個(gè)變量名進(jìn)行重構(gòu)后小泉,Qt似乎重構(gòu)了好久,等重構(gòu)完成后臀防,發(fā)現(xiàn)這個(gè)Qt的構(gòu)建套件已經(jīng)報(bào)廢了略板,無(wú)論編譯構(gòu)建什么工程毁枯,都會(huì)報(bào)錯(cuò),而且錯(cuò)誤來(lái)源都是Qt自己的頭文件叮称。
后來(lái)我也是折騰了半天种玛,百思不得其解胀糜,懷疑是那次重構(gòu)Qt把自己的內(nèi)部頭文件里的變量名也給重構(gòu)了(內(nèi)部頭文件中存在變量剛好和我要重構(gòu)的這個(gè)變量名相同)。后來(lái)我重新下載了一個(gè)Qt蒂誉,手動(dòng)把那個(gè)敏感的變量重新命名了一下,問(wèn)題就解決了距帅。
QAbstractButton的類型轉(zhuǎn)換問(wèn)題
實(shí)例化一個(gè)對(duì)話框QMessageBox右锨,添加自定義的按鈕。代碼寫(xiě)法如下:
QMessageBox box;
box.setWindowTitle("標(biāo)題");
box.setText("文字");
QPushButton *connectButton = box.addButton(tr("確定"), QMessageBox::ActionRole);
QPushButton *abortButton = box.addButton(tr("取消"),QMessageBox::ActionRole);
box.exec();
QAbstractButton* btnClicked = box.clickedButton();
QPushButton* btn = dynamic_cast<QPushButton*>(box.clickedButton());
QPushButton* btn = qobject_cast<QPushButton*>(box.clickedButton());
if (btn == connectButton){
this->close();
} else if (btn == abortButton){
;
}
但是實(shí)際上QMessageBox的clickButton()方法返回的是QAbstractButton*類型碌秸,因此需要進(jìn)行類型轉(zhuǎn)換绍移,因?yàn)镼AbstractButton是QPushButton的基類,因此我用的是dynamic_cast讥电。然而我驚訝地發(fā)現(xiàn)這樣寫(xiě)會(huì)報(bào)錯(cuò):
然后我又試了一下qobject_cast蹂窖,也不行,最后直接用(QPushButton*)ptr 強(qiáng)轉(zhuǎn)了恩敌。我至今也沒(méi)搞懂為什么無(wú)法進(jìn)行轉(zhuǎn)換瞬测。
不穩(wěn)定的插件
Qt里面有一個(gè)Fake Vim,我就試用了一下纠炮。那一次打算在頭文件里輸入一個(gè) #include 語(yǔ)句月趟,結(jié)果忘了自己在用Vim,習(xí)慣性地打出了第一個(gè)字符 # 恢口。 當(dāng)時(shí)Vim不在編輯模式孝宗,按理來(lái)說(shuō)對(duì)這個(gè)輸入應(yīng)該是沒(méi)有響應(yīng)的,結(jié)果Qt就崩潰了耕肩,沒(méi)有保存的修改也丟失了因妇。
自從那次之后,我就不敢用Qt的Fake Vim了猿诸。那還是一年多前的事婚被,現(xiàn)在Qt也更新了好多版本,不知道問(wèn)題修復(fù)了沒(méi)有两芳。
動(dòng)態(tài)補(bǔ)充中
其實(shí)應(yīng)該還有不少奇怪的問(wèn)題摔寨,不過(guò)已經(jīng)記不太清楚了,下次遇到再補(bǔ)充到Qt里來(lái)怖辆。
其實(shí)從以上的文字看我似乎在吐槽Qt是复,但其實(shí)我個(gè)人還是比較喜歡用Qt的。IDE用起來(lái)還是比較順手的竖螃,至少比VS要順手得多淑廊。Qt的信號(hào)-槽機(jī)制我也很喜歡,很多類之間通信的代碼寫(xiě)起來(lái)方便多了特咆,不用寫(xiě)一堆麻煩的回調(diào)季惩。