Qt入門(10)——除錯技術
命令列引數
當你執行Qt程式時,你可以指定幾個命令列引數來幫助你除錯。
-nograb 應用程式不再捕獲滑鼠或者鍵盤。當程式在Linux下執行在gdb偵錯程式中時這個選項是預設的。
-dograb 忽略任何隱含的或明顯得-nograb。即使-nograb出現在命令列的最後,-dograb也會超過-nograb生效的。
-sync 在X同步模式下執行應用程式。同步模式強迫X伺服器立即執行每一個X客戶端的請求,而並不能使用快取優化。它使得程式更加容易測試並且通常會更慢。-sync模式只對X11版本的Qt有效。
警告和除錯訊息
Qt包含了三個全域性函式來寫出警告和除錯文字。
qDebug()用來為測試及其它寫除錯輸出。
qWarning()用來在程式發生錯誤時寫警告輸出。
qFatal()用來寫致命錯誤訊息並且退出。
這些函式的Qt實現是在Unix/X11下把文字列印到標準錯誤(stderr)輸出,在Windows下會列印到偵錯程式。你可以通過安裝一個訊息處理器,qInstallMsgHandler()來接收這些函式。
當應用程式看起來或者用起來很奇怪的時候,除錯函式QObject::dumpObjectTree()和QObject::dumpObjectInfo()很有用。如果你使用了物件名稱,這將會更有用,但通常情況下是沒有名稱的。
除錯巨集
qglobal.h標頭檔案包含了很多除錯巨集和#defines。
兩個重要的巨集是:
Q_ASSERT(b)裡面的b是一個布林表示式,當b是FALSE的時候,寫出警告資訊:“ASSERT: 'b' in file file.cpp (234)”。
Q_CHECK_PTR(p)裡面的p是一個指標。如果p是空的話,寫出警告資訊:“In file file.cpp, line 234: Out of memory”。
這些巨集在檢測程式錯誤時很有用,比如像這樣:
char *alloc( int size )
{
Q_ASSERT( size > 0 );
char *p = new char[size];
Q_CHECK_PTR( p );
return p;
}
如果你定義了QT_FATAL_ASSERT標記,Q_ASSERT將會呼叫fatal()而不是warning(),所以一個錯誤宣告將會導致在列印錯誤訊息後使程式退出。
注意如果QT_CHECK_STATE未定義,Q_ASSERT巨集就是一個空的表示式(參見下面)。在裡面的任何程式碼都不會被執行。相似的,如果QT_CHECK_NULL未定義,Q_CHECK_PTR也是一個空的表示式。這裡就是一個不應該如此使用Q_ASSERT和Q_CHECK_PTR的例子:
char *alloc( int size )
{
char *p;
Q_CHECK_PTR( p = new char[size] ); // WRONG
return p;
}
這個問題是棘手的:僅僅在正確的檢測標記被定義時,p才會被設定為健全的值。如果QT_CHECK_NULL標記沒有被定義,程式碼被編譯了,在Q_CHECK_PTR表示式中的程式碼是不會被執行的(正確地,因為它僅僅用於除錯目的)並且會分配一個瘋狂的指標。
Qt庫包含了幾百個內部檢查,當一些錯誤被檢測出時,會列印警告資訊。
Qt中的健全測試和作為結果的警告資訊是有條件的,基於不同的除錯標記的狀態:
QT_CHECK_STATE:檢測一致的/期望的物件狀態
QT_CHECK_RANGE:檢測變數範圍錯誤
QT_CHECK_NULL:檢測危險的空指標
QT_CHECK_MATH:檢測危險的數學,比如被0除
QT_NO_CHECK:關閉所有的QT_CHECK_...標記
QT_DEBUG:使除錯程式碼生效
QT_NO_DEBUG:關閉QT_DEBUG標記
預設情況下,QT_DEBUG和所有的QT_CHECK標記都是開啟的。如果要關閉QT_DEBUG,請定義QT_NO_DEBUG。如果要關閉QT_CHECK標記,請定義QT_NO_CHECK。
例項:
void f( char *p, int i )
{
#if defined(QT_CHECK_NULL)
if ( p == 0 )
qWarning( "f: Null pointer not allowed" );
#endif
#if defined(QT_CHECK_RANGE)
if ( i < 0 )
qWarning( "f: The index cannot be negative" );
#endif
}
普通的Bug
這是一個如此普通的bug,所以要在這裡提到:如果你你在類生命中包含了Q_OBJECT巨集並且執行了moc,但是忘記了把moc生成的物件程式碼連線到你的可執行程式中,你就會得到非常困惑的錯誤訊息。
任何一個會提示缺乏vtbl、_vtbl、__vtbl或者和這個類似的連線錯誤都有可能是這樣的問題。
相關文章
- 前端除錯入門前端除錯
- Kafka除錯入門(一)Kafka除錯
- .NET高階除錯系列-Windbg除錯入門篇高階除錯
- Google Chrome除錯js入門GoChrome除錯JS
- Node 除錯工具入門教程除錯
- Web除錯技術詳解Web除錯
- QT入門QT
- QT release 生成除錯符合QT除錯
- Xcode Instruments除錯swift入門教程XCode除錯Swift
- 深入Java除錯技術 -Ari NomanJava除錯
- QT快速入門QT
- NodeMCU入門:燒錄、除錯、聯網除錯
- GDB除錯-從入門到實踐除錯
- 前端入門技巧之瀏覽器除錯前端瀏覽器除錯
- 程式碼除錯-入門、實踐到原理除錯
- Qt Quick 如何入門?QTUI
- Qt入門之概述QT
- Golang反射技術初始入門Golang反射
- Mqtt入門:線上除錯連線阿里雲MQQT除錯阿里
- Elasticsearch核心技術(二):Elasticsearch入門Elasticsearch
- Windows 反除錯技術——OpenProcess 許可權過濾Windows除錯
- AI考拉技術分享–Scrum入門AIScrum
- 區塊鏈技術入門應用區塊鏈
- AI考拉技術分享--Scrum入門AIScrum
- 容器技術之Docker基礎入門Docker
- 入門web前端需要掌握的技術Web前端
- MySQL入門--複製技術介紹MySql
- SpringCloud微服務治理技術入門(SCN)SpringGCCloud微服務
- Java技術分享:小白如何入門Mybatis?JavaMyBatis
- win10 如何除錯串列埠_win10串列埠除錯怎麼除錯Win10除錯串列埠
- Flutter入門——山寨掘金(一)| 掘金技術徵文Flutter
- Docker技術筆記:Docker入門淺嘗Docker筆記
- JVM效能優化(一)JVM技術入門JVM優化
- 入門 Web前端需要學習哪些技術?Web前端
- Android應用方法隱藏及反除錯技術淺析Android除錯
- qt中debug顯示64位不能除錯32位QT除錯
- 實時音視訊技術入門提綱
- [譯]區塊鏈技術全解析入門版區塊鏈
- web專案技術必備-------jQuery快速入門WebjQuery