亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

VC調(diào)試以及TRACE()函數(shù)的用法

系統(tǒng) 2550 0

1. VC調(diào)試入門

設(shè)置
為了調(diào)試一個(gè)程序,首先必須使程序中包含調(diào)試信息。一般情況下,一個(gè)從AppWizard創(chuàng)建的工程中包含的Debug Configuration自動(dòng)包含調(diào)試信息,但是是不是Debug版本并不是程序包含調(diào)試信息的決定因素,程序設(shè)計(jì)者可以在任意的Configuration中增加調(diào)試信息,包括Release版本。
為了增加調(diào)試信息,可以按照下述步驟進(jìn)行:

  • 打開Project settings對(duì)話框(可以通過快捷鍵ALT+F7打開,也可以通過IDE菜單Project/Settings打開)
  • 選擇C/C++頁,Category中選擇general ,則出現(xiàn)一個(gè)Debug Info下拉列表框,可供選擇的調(diào)試信息 方式包括:
    命令行 Project settings 說明
    None 沒有調(diào)試信息
    /Zd Line Numbers Only 目標(biāo)文件或者可執(zhí)行文件中只包含全局和導(dǎo)出符號(hào)以及代碼行信息,不包含符號(hào)調(diào)試信息
    /Z7 C 7.0- Compatible 目標(biāo)文件或者可執(zhí)行文件中包含行號(hào)和所有符號(hào)調(diào)試信息,包括變量名及類型,函數(shù)及原型等
    /Zi Program Database 創(chuàng)建一個(gè)程序庫(PDB),包括類型信息和符號(hào)調(diào)試信息。
    /ZI Program Database for Edit and Continue 除了前面/Zi的功能外,這個(gè)選項(xiàng)允許對(duì)代碼進(jìn)行調(diào)試過程中的修改和繼續(xù)執(zhí)行。這個(gè)選項(xiàng)同時(shí)使#pragma設(shè)置的優(yōu)化功能無效
  • 選擇Link頁,選中復(fù)選框"Generate Debug Info",這個(gè)選項(xiàng)將使連接器把調(diào)試信息寫進(jìn)可執(zhí)行文件和DLL
  • 如果C/C++頁中設(shè)置了Program Database以上的選項(xiàng),則Link incrementally可以選擇。選中這個(gè)選項(xiàng),將使程序可以在上一次編譯的基礎(chǔ)上被編譯(即增量編譯),而不必每次都從頭開始編譯。

設(shè)置
為了調(diào)試一個(gè)程序,首先必須使程序中包含調(diào)試信息。一般情況下,一個(gè)從AppWizard創(chuàng)建的工程中包含的Debug Configuration自動(dòng)包含調(diào)試信息,但是是不是Debug版本并不是程序包含調(diào)試信息的決定因素,程序設(shè)計(jì)者可以在任意的Configuration中增加調(diào)試信息,包括Release版本。
為了增加調(diào)試信息,可以按照下述步驟進(jìn)行:

  • 打開Project settings對(duì)話框(可以通過快捷鍵ALT+F7打開,也可以通過IDE菜單Project/Settings打開)
  • 選擇C/C++頁,Category中選擇general ,則出現(xiàn)一個(gè)Debug Info下拉列表框,可供選擇的調(diào)試信息 方式包括:
    命令行 Project settings 說明
    None 沒有調(diào)試信息
    /Zd Line Numbers Only 目標(biāo)文件或者可執(zhí)行文件中只包含全局和導(dǎo)出符號(hào)以及代碼行信息,不包含符號(hào)調(diào)試信息
    /Z7 C 7.0- Compatible 目標(biāo)文件或者可執(zhí)行文件中包含行號(hào)和所有符號(hào)調(diào)試信息,包括變量名及類型,函數(shù)及原型等
    /Zi Program Database 創(chuàng)建一個(gè)程序庫(PDB),包括類型信息和符號(hào)調(diào)試信息。
    /ZI Program Database for Edit and Continue 除了前面/Zi的功能外,這個(gè)選項(xiàng)允許對(duì)代碼進(jìn)行調(diào)試過程中的修改和繼續(xù)執(zhí)行。這個(gè)選項(xiàng)同時(shí)使#pragma設(shè)置的優(yōu)化功能無效
  • 選擇Link頁,選中復(fù)選框"Generate Debug Info",這個(gè)選項(xiàng)將使連接器把調(diào)試信息寫進(jìn)可執(zhí)行文件和DLL
  • 如果C/C++頁中設(shè)置了Program Database以上的選項(xiàng),則Link incrementally可以選擇。選中這個(gè)選項(xiàng),將使程序可以在上一次編譯的基礎(chǔ)上被編譯(即增量編譯),而不必每次都從頭開始編譯。

斷點(diǎn)
斷點(diǎn)是調(diào)試器設(shè)置的一個(gè)代碼位置。當(dāng)程序運(yùn)行到斷點(diǎn)時(shí),程序中斷執(zhí)行,回到調(diào)試器。斷點(diǎn)是 最常用的技巧。調(diào)試時(shí),只有設(shè)置了斷點(diǎn)并使程序回到調(diào)試器,才能對(duì)程序進(jìn)行在線調(diào)試。

設(shè)置斷點(diǎn):可以通過下述方法設(shè)置一個(gè)斷點(diǎn)。首先把光標(biāo)移動(dòng)到需要設(shè)置斷點(diǎn)的代碼行上,然后

  • 按F9快捷鍵
  • 彈出Breakpoints對(duì)話框,方法是按快捷鍵CTRL+B或ALT+F9,或者通過菜單Edit/Breakpoints打開。打開后點(diǎn)擊Break at編輯框的右側(cè)的箭頭,選擇 合適的位置信息。一般情況下,直接選擇line xxx就足夠了,如果想設(shè)置不是當(dāng)前位置的斷點(diǎn),可以選擇Advanced,然后填寫函數(shù)、行號(hào)和可執(zhí)行文件信息。

去掉斷點(diǎn):把光標(biāo)移動(dòng)到給定斷點(diǎn)所在的行,再次按F9就可以取消斷點(diǎn)。同前面所述,打開Breakpoints對(duì)話框后,也可以按照界面提示去掉斷點(diǎn)。

條件斷點(diǎn):可以為斷點(diǎn)設(shè)置一個(gè)條件,這樣的斷點(diǎn)稱為條件斷點(diǎn)。對(duì)于新加的斷點(diǎn),可以單擊Conditions按鈕,為斷點(diǎn)設(shè)置一個(gè)表達(dá)式。當(dāng)這個(gè)表達(dá)式發(fā)生改變時(shí),程序就 被中斷。底下設(shè)置包括“觀察數(shù)組或者結(jié)構(gòu)的元素個(gè)數(shù)”,似乎可以設(shè)置一個(gè)指針?biāo)赶虻膬?nèi)存區(qū)的大小,但是我設(shè)置一個(gè)比較的值但是改動(dòng) 范圍之外的內(nèi)存區(qū)似乎也導(dǎo)致斷點(diǎn)起效。最后一個(gè)設(shè)置可以讓程序先執(zhí)行多少次然后才到達(dá)斷點(diǎn)。

數(shù)據(jù)斷點(diǎn):數(shù)據(jù)斷點(diǎn)只能在Breakpoints對(duì)話框中設(shè)置。選擇“Data”頁,就顯示了設(shè)置數(shù)據(jù)斷點(diǎn)的對(duì)話框。在編輯框中輸入一個(gè)表達(dá)式,當(dāng)這個(gè) 表達(dá)式的值發(fā)生變化時(shí),數(shù)據(jù)斷點(diǎn)就到達(dá)。一般情況下,這個(gè)表達(dá)式應(yīng)該由運(yùn)算符和全局變量構(gòu)成,例如:在編輯框中輸入 g_bFlag這個(gè)全局變量的名字,那么當(dāng)程序中有g(shù)_bFlag= !g_bFlag時(shí),程序就將停在這個(gè)語句處。

消息斷點(diǎn):VC也支持對(duì)Windows消息進(jìn)行截獲。他有兩種方式進(jìn)行截獲:窗口消息處理函數(shù)和特定消息中斷。
在Breakpoints對(duì)話框中選擇Messages頁,就可以設(shè)置消息斷點(diǎn)。如果在上面那個(gè)對(duì)話框中寫入消息處理函數(shù)的名字,那么 每次消息被這個(gè)函數(shù)處理,斷點(diǎn)就到達(dá)(我覺得如果采用普通斷點(diǎn)在這個(gè)函數(shù)中截獲,效果應(yīng)該一樣)。如果在底下的下拉 列表框選擇一個(gè)消息,則每次這種消息到達(dá),程序就中斷。


Watch
VC支持查看變量、表達(dá)式和內(nèi)存的值。所有這些觀察都必須是在斷點(diǎn)中斷的情況下進(jìn)行。
觀看變量的值最簡(jiǎn)單,當(dāng)斷點(diǎn)到達(dá)時(shí),把光標(biāo)移動(dòng)到這個(gè)變量上,停留一會(huì)就可以看到變量的值。
VC提供一種被成為Watch的機(jī)制來觀看變量和表達(dá)式的值。在斷點(diǎn)狀態(tài)下,在變量上單擊右鍵,選擇Quick Watch, 就彈出一個(gè)對(duì)話框,顯示這個(gè)變量的值。
單擊Debug工具條上的Watch按鈕,就出現(xiàn)一個(gè)Watch視圖(Watch1,Watch2,Watch3,Watch4),在該視圖中輸入變量或者表達(dá)式,就可以觀察 變量或者表達(dá)式的值。注意:這個(gè)表達(dá)式不能有副作用,例如++運(yùn)算符絕對(duì)禁止用于這個(gè)表達(dá)式中,因?yàn)檫@個(gè)運(yùn)算符將修改變量的值,導(dǎo)致 軟件的邏輯被破壞。

Memory
由于指針指向的數(shù)組,Watch只能顯示第一個(gè)元素的值。為了顯示數(shù)組的后續(xù)內(nèi)容,或者要顯示一片內(nèi)存的內(nèi)容,可以使用memory功能。在 Debug工具條上點(diǎn)memory按鈕,就彈出一個(gè)對(duì)話框,在其中輸入地址,就可以顯示該地址指向的內(nèi)存的內(nèi)容。


Varibles

Debug工具條上的Varibles按鈕彈出一個(gè)框,顯示所有當(dāng)前執(zhí)行上下文中可見的變量的值。特別是當(dāng)前指令涉及的變量,以紅色顯示。

寄存器
Debug工具條上的Reigsters按鈕彈出一個(gè)框,顯示當(dāng)前的所有寄存器的值。

進(jìn)程控制
VC允許被中斷的程序繼續(xù)運(yùn)行、單步運(yùn)行和運(yùn)行到指定光標(biāo)處,分別對(duì)應(yīng)快捷鍵F5、F10/F11和CTRL+F10。各個(gè)快捷鍵功能如下:

快捷鍵 說明
F5 繼續(xù)運(yùn)行
F10 單步,如果涉及到子函數(shù),不進(jìn)入子函數(shù)內(nèi)部
F11 單步,如果涉及到子函數(shù),進(jìn)入子函數(shù)內(nèi)部
CTRL+F10 運(yùn)行到當(dāng)前光標(biāo)處。

Call Stack
調(diào)用堆棧反映了當(dāng)前斷點(diǎn)處函數(shù)是被那些函數(shù)按照什么順序調(diào)用的。單擊Debug工具條上的Call stack就顯示Call Stack對(duì)話框。在CallStack對(duì)話框中顯示了一個(gè)調(diào)用系列,最上面的是當(dāng)前函數(shù),往下依次是調(diào)用函數(shù)的上級(jí)函數(shù)。單擊這些函數(shù)名可以跳到對(duì)應(yīng)的函數(shù)中去。

其他調(diào)試手段
系統(tǒng)提供一系列特殊的函數(shù)或者宏來處理Debug版本相關(guān)的信息,如下:

宏名/函數(shù)名 說明
TRACE 使用方法和printf完全一致,他在output框中輸出調(diào)試信息
ASSERT 它接收一個(gè)表達(dá)式,如果這個(gè)表達(dá)式為TRUE,則無動(dòng)作,否則中斷當(dāng)前程序執(zhí)行。對(duì)于系統(tǒng)中出現(xiàn)這個(gè)宏 導(dǎo)致的中斷,應(yīng)該認(rèn)為你的函數(shù)調(diào)用未能滿足系統(tǒng)的調(diào)用此函數(shù)的前提條件。例如,對(duì)于一個(gè)還沒有創(chuàng)建的窗口調(diào)用SetWindowText等。
VERIFY 和ASSERT功能類似,所不同的是,在Release版本中,ASSERT不計(jì)算輸入的表達(dá)式的值,而VERIFY計(jì)算表達(dá)式的值。

關(guān)注
一個(gè)好的程序員不應(yīng)該把所有的判斷交給編譯器和調(diào)試器,應(yīng)該在程序中自己加以程序保護(hù)和錯(cuò)誤定位,具體措施包括:

  • 對(duì)于所有有返回值的函數(shù),都應(yīng)該檢查返回值,除非你確信這個(gè)函數(shù)調(diào)用絕對(duì)不會(huì)出錯(cuò),或者不關(guān)心它是否出錯(cuò)。
  • 一些函數(shù)返回錯(cuò)誤,需要用其他函數(shù)獲得錯(cuò)誤的具體信息。例如accept返回INVALID_SOCKET表示accept失敗,為了查明 具體的失敗原因,應(yīng)該立刻用WSAGetLastError獲得錯(cuò)誤碼,并針對(duì)性的解決問題。
  • 有些函數(shù)通過異常機(jī)制拋出錯(cuò)誤,應(yīng)該用TRY-CATCH語句來檢查錯(cuò)誤
  • 程序員對(duì)于能處理的錯(cuò)誤,應(yīng)該自己在底層處理,對(duì)于不能處理的,應(yīng)該報(bào)告給用戶讓他們決定怎么處理。如果程序出了異常, 卻不對(duì)返回值和其他機(jī)制返回的錯(cuò)誤信息進(jìn)行判斷,只能是加大了找錯(cuò)誤的難度。

另外:VC中要編制程序不應(yīng)該一開始就寫cpp/h文件,而應(yīng)該首先創(chuàng)建一個(gè)合適的工程。因?yàn)橹挥羞@樣,VC才能選擇合適的編譯、連接 選項(xiàng)。對(duì)于加入到工程中的cpp文件,應(yīng)該檢查是否在第一行顯式的包含stdafx.h頭文件,這是Microsoft Visual Studio為了加快編譯 速度而設(shè)置的預(yù)編譯頭文件。在這個(gè)#include "stdafx.h"行前面的所有代碼將被忽略,所以其他頭文件應(yīng)該在這一行后面被包含。
對(duì)于.c文件,由于不能包含stdafx.h,因此可以通過Project settings把它的預(yù)編譯頭設(shè)置為“不使用”,方法是:

  • 彈出Project settings對(duì)話框
  • 選擇C/C++
  • Category選擇Precompilation Header
  • 選擇不使用預(yù)編譯頭。

斷點(diǎn)
斷點(diǎn)是調(diào)試器設(shè)置的一個(gè)代碼位置。當(dāng)程序運(yùn)行到斷點(diǎn)時(shí),程序中斷執(zhí)行,回到調(diào)試器。斷點(diǎn)是 最常用的技巧。調(diào)試時(shí),只有設(shè)置了斷點(diǎn)并使程序回到調(diào)試器,才能對(duì)程序進(jìn)行在線調(diào)試。

設(shè)置斷點(diǎn):可以通過下述方法設(shè)置一個(gè)斷點(diǎn)。首先把光標(biāo)移動(dòng)到需要設(shè)置斷點(diǎn)的代碼行上,然后

  • 按F9快捷鍵
  • 彈出Breakpoints對(duì)話框,方法是按快捷鍵CTRL+B或ALT+F9,或者通過菜單Edit/Breakpoints打開。打開后點(diǎn)擊Break at編輯框的右側(cè)的箭頭,選擇 合適的位置信息。一般情況下,直接選擇line xxx就足夠了,如果想設(shè)置不是當(dāng)前位置的斷點(diǎn),可以選擇Advanced,然后填寫函數(shù)、行號(hào)和可執(zhí)行文件信息。

去掉斷點(diǎn):把光標(biāo)移動(dòng)到給定斷點(diǎn)所在的行,再次按F9就可以取消斷點(diǎn)。同前面所述,打開Breakpoints對(duì)話框后,也可以按照界面提示去掉斷點(diǎn)。

條件斷點(diǎn):可以為斷點(diǎn)設(shè)置一個(gè)條件,這樣的斷點(diǎn)稱為條件斷點(diǎn)。對(duì)于新加的斷點(diǎn),可以單擊Conditions按鈕,為斷點(diǎn)設(shè)置一個(gè)表達(dá)式。當(dāng)這個(gè)表達(dá)式發(fā)生改變時(shí),程序就 被中斷。底下設(shè)置包括“觀察數(shù)組或者結(jié)構(gòu)的元素個(gè)數(shù)”,似乎可以設(shè)置一個(gè)指針?biāo)赶虻膬?nèi)存區(qū)的大小,但是我設(shè)置一個(gè)比較的值但是改動(dòng) 范圍之外的內(nèi)存區(qū)似乎也導(dǎo)致斷點(diǎn)起效。最后一個(gè)設(shè)置可以讓程序先執(zhí)行多少次然后才到達(dá)斷點(diǎn)。

數(shù)據(jù)斷點(diǎn):數(shù)據(jù)斷點(diǎn)只能在Breakpoints對(duì)話框中設(shè)置。選擇“Data”頁,就顯示了設(shè)置數(shù)據(jù)斷點(diǎn)的對(duì)話框。在編輯框中輸入一個(gè)表達(dá)式,當(dāng)這個(gè) 表達(dá)式的值發(fā)生變化時(shí),數(shù)據(jù)斷點(diǎn)就到達(dá)。一般情況下,這個(gè)表達(dá)式應(yīng)該由運(yùn)算符和全局變量構(gòu)成,例如:在編輯框中輸入 g_bFlag這個(gè)全局變量的名字,那么當(dāng)程序中有g(shù)_bFlag= !g_bFlag時(shí),程序就將停在這個(gè)語句處。

消息斷點(diǎn):VC也支持對(duì)Windows消息進(jìn)行截獲。他有兩種方式進(jìn)行截獲:窗口消息處理函數(shù)和特定消息中斷。
在Breakpoints對(duì)話框中選擇Messages頁,就可以設(shè)置消息斷點(diǎn)。如果在上面那個(gè)對(duì)話框中寫入消息處理函數(shù)的名字,那么 每次消息被這個(gè)函數(shù)處理,斷點(diǎn)就到達(dá)(我覺得如果采用普通斷點(diǎn)在這個(gè)函數(shù)中截獲,效果應(yīng)該一樣)。如果在底下的下拉 列表框選擇一個(gè)消息,則每次這種消息到達(dá),程序就中斷。


Watch
VC支持查看變量、表達(dá)式和內(nèi)存的值。所有這些觀察都必須是在斷點(diǎn)中斷的情況下進(jìn)行。
觀看變量的值最簡(jiǎn)單,當(dāng)斷點(diǎn)到達(dá)時(shí),把光標(biāo)移動(dòng)到這個(gè)變量上,停留一會(huì)就可以看到變量的值。
VC提供一種被成為Watch的機(jī)制來觀看變量和表達(dá)式的值。在斷點(diǎn)狀態(tài)下,在變量上單擊右鍵,選擇Quick Watch, 就彈出一個(gè)對(duì)話框,顯示這個(gè)變量的值。
單擊Debug工具條上的Watch按鈕,就出現(xiàn)一個(gè)Watch視圖(Watch1,Watch2,Watch3,Watch4),在該視圖中輸入變量或者表達(dá)式,就可以觀察 變量或者表達(dá)式的值。注意:這個(gè)表達(dá)式不能有副作用,例如++運(yùn)算符絕對(duì)禁止用于這個(gè)表達(dá)式中,因?yàn)檫@個(gè)運(yùn)算符將修改變量的值,導(dǎo)致 軟件的邏輯被破壞。

Memory
由于指針指向的數(shù)組,Watch只能顯示第一個(gè)元素的值。為了顯示數(shù)組的后續(xù)內(nèi)容,或者要顯示一片內(nèi)存的內(nèi)容,可以使用memory功能。在 Debug工具條上點(diǎn)memory按鈕,就彈出一個(gè)對(duì)話框,在其中輸入地址,就可以顯示該地址指向的內(nèi)存的內(nèi)容。


Varibles

Debug工具條上的Varibles按鈕彈出一個(gè)框,顯示所有當(dāng)前執(zhí)行上下文中可見的變量的值。特別是當(dāng)前指令涉及的變量,以紅色顯示。

寄存器
Debug工具條上的Reigsters按鈕彈出一個(gè)框,顯示當(dāng)前的所有寄存器的值。

進(jìn)程控制
VC允許被中斷的程序繼續(xù)運(yùn)行、單步運(yùn)行和運(yùn)行到指定光標(biāo)處,分別對(duì)應(yīng)快捷鍵F5、F10/F11和CTRL+F10。各個(gè)快捷鍵功能如下:

快捷鍵 說明
F5 繼續(xù)運(yùn)行
F10 單步,如果涉及到子函數(shù),不進(jìn)入子函數(shù)內(nèi)部
F11 單步,如果涉及到子函數(shù),進(jìn)入子函數(shù)內(nèi)部
CTRL+F10 運(yùn)行到當(dāng)前光標(biāo)處。

Call Stack
調(diào)用堆棧反映了當(dāng)前斷點(diǎn)處函數(shù)是被那些函數(shù)按照什么順序調(diào)用的。單擊Debug工具條上的Call stack就顯示Call Stack對(duì)話框。在CallStack對(duì)話框中顯示了一個(gè)調(diào)用系列,最上面的是當(dāng)前函數(shù),往下依次是調(diào)用函數(shù)的上級(jí)函數(shù)。單擊這些函數(shù)名可以跳到對(duì)應(yīng)的函數(shù)中去。

其他調(diào)試手段
系統(tǒng)提供一系列特殊的函數(shù)或者宏來處理Debug版本相關(guān)的信息,如下:

宏名/函數(shù)名 說明
TRACE 使用方法和printf完全一致,他在output框中輸出調(diào)試信息
ASSERT 它接收一個(gè)表達(dá)式,如果這個(gè)表達(dá)式為TRUE,則無動(dòng)作,否則中斷當(dāng)前程序執(zhí)行。對(duì)于系統(tǒng)中出現(xiàn)這個(gè)宏 導(dǎo)致的中斷,應(yīng)該認(rèn)為你的函數(shù)調(diào)用未能滿足系統(tǒng)的調(diào)用此函數(shù)的前提條件。例如,對(duì)于一個(gè)還沒有創(chuàng)建的窗口調(diào)用SetWindowText等。
VERIFY 和ASSERT功能類似,所不同的是,在Release版本中,ASSERT不計(jì)算輸入的表達(dá)式的值,而VERIFY計(jì)算表達(dá)式的值。

關(guān)注
一個(gè)好的程序員不應(yīng)該把所有的判斷交給編譯器和調(diào)試器,應(yīng)該在程序中自己加以程序保護(hù)和錯(cuò)誤定位,具體措施包括:

  • 對(duì)于所有有返回值的函數(shù),都應(yīng)該檢查返回值,除非你確信這個(gè)函數(shù)調(diào)用絕對(duì)不會(huì)出錯(cuò),或者不關(guān)心它是否出錯(cuò)。
  • 一些函數(shù)返回錯(cuò)誤,需要用其他函數(shù)獲得錯(cuò)誤的具體信息。例如accept返回INVALID_SOCKET表示accept失敗,為了查明 具體的失敗原因,應(yīng)該立刻用WSAGetLastError獲得錯(cuò)誤碼,并針對(duì)性的解決問題。
  • 有些函數(shù)通過異常機(jī)制拋出錯(cuò)誤,應(yīng)該用TRY-CATCH語句來檢查錯(cuò)誤
  • 程序員對(duì)于能處理的錯(cuò)誤,應(yīng)該自己在底層處理,對(duì)于不能處理的,應(yīng)該報(bào)告給用戶讓他們決定怎么處理。如果程序出了異常, 卻不對(duì)返回值和其他機(jī)制返回的錯(cuò)誤信息進(jìn)行判斷,只能是加大了找錯(cuò)誤的難度。

另外:VC中要編制程序不應(yīng)該一開始就寫cpp/h文件,而應(yīng)該首先創(chuàng)建一個(gè)合適的工程。因?yàn)橹挥羞@樣,VC才能選擇合適的編譯、連接 選項(xiàng)。對(duì)于加入到工程中的cpp文件,應(yīng)該檢查是否在第一行顯式的包含stdafx.h頭文件,這是Microsoft Visual Studio為了加快編譯 速度而設(shè)置的預(yù)編譯頭文件。在這個(gè)#include "stdafx.h"行前面的所有代碼將被忽略,所以其他頭文件應(yīng)該在這一行后面被包含。
對(duì)于.c文件,由于不能包含stdafx.h,因此可以通過Project settings把它的預(yù)編譯頭設(shè)置為“不使用”,方法是:

  • 彈出Project settings對(duì)話框
  • 選擇C/C++
  • Category選擇Precompilation Header
  • 選擇不使用預(yù)編譯頭。

2. 調(diào)試輸出說明

在調(diào)試程序的時(shí)候,我們可以使用下面這3個(gè)函數(shù)將調(diào)試信息輸出到控制臺(tái)窗口,這對(duì)我們調(diào)試程序非常有幫助:

            
              echo(text);
warn(text);
error(text);

            
          

其中echo函數(shù)用于輸出標(biāo)準(zhǔn)黑色字體的文本信息到控制臺(tái)窗口中;warn函數(shù)用于輸出標(biāo)準(zhǔn)灰色字體的文本信息到控制臺(tái)窗口中;error函數(shù)用于輸出標(biāo)準(zhǔn)紅色字體的文本信息到控制臺(tái)窗口中。使用上述3個(gè)函數(shù)的時(shí)候,文本可根據(jù)字符串規(guī)則進(jìn)行格式化。

通過正確地使用恰當(dāng)?shù)妮敵鱿ⅲ梢愿櫾谀_本中發(fā)生的任何事件。一般情況下,人們都希望把最可能出現(xiàn)問題的地方的相關(guān)調(diào)試信息輸出到控制臺(tái)。您或許想將一個(gè)容易識(shí)別的標(biāo)志符置于代碼某處,以便在控制臺(tái)的滾動(dòng)窗口中找到它;或者輸出一些與代碼相關(guān)的活動(dòng)的重要信息(這些信息可能是一些變量的輸出)。

為了把控制臺(tái)輸出信息轉(zhuǎn)儲(chǔ)在console.log文件中,首先,您必須在您的代碼中調(diào)用setLogMode函數(shù),并確保該函數(shù)在程序運(yùn)行至有問題的代碼語句之前被執(zhí)行。一個(gè)更為簡(jiǎn)單的辦法就是:使用-log命令行選項(xiàng),后接一個(gè)空格,然后輸入數(shù)字0、1或2。其中0表示您不能記錄日志;1表示您可以把每一個(gè)新會(huì)話的內(nèi)容追加到日志文件的末尾;2表示您可以用新日志內(nèi)容覆蓋以前的日志文件。

            
              您可以用下面的方式在程序中設(shè)置標(biāo)記:
error("******************************************");
它將在console . log文件中添加一行星號(hào):
******************************************;

            
          

在控制臺(tái)窗口中,這行星號(hào)會(huì)以紅色字符顯示,因此很容易識(shí)別。您也可以使用warn函數(shù)向控制臺(tái)窗口輸出灰色字符或使用echo函數(shù)向控制臺(tái)窗口輸出黑色字符。

很多時(shí)候,如果游戲中出現(xiàn)停頓或者死鎖的情況,那么,即使最后一行代碼已經(jīng)被執(zhí)行,它的調(diào)試輸出信息也不會(huì)被寫到日志文件中。處理這種情況的方法很簡(jiǎn)單:在程序中設(shè)置兩個(gè)一樣的錯(cuò)誤行,一個(gè)緊接著位于另一個(gè)的右端。如果游戲暫停,則只有第一行消息輸出,那么此時(shí)您就可以縮小錯(cuò)誤查找的范圍。接下來,把這兩個(gè)標(biāo)記行往程序下方移動(dòng),直到它們不再出現(xiàn)在控制臺(tái)日志中。這時(shí),就可以找到出現(xiàn)問題的代碼位置。

如果需要檢查一些重要變量的值,比如說X,Y和玩家的名稱,您可能會(huì)使用如下語句:

            
              echo("player’s name: " @ playerName @ " X= " %X @ " Y=" @%Y );
            
          
然后,可以在控制臺(tái)窗口的日志末尾看到這樣的輸出信息:
            
              player’s name: bozotheclown X=123 Y=456
            
          


3. 使用trace函數(shù)

Torque提供了一個(gè)很方便的跟蹤函數(shù)trace。使用它,在執(zhí)行腳本文件時(shí)就可以計(jì)算出當(dāng)前正在執(zhí)行的是哪一行代碼。當(dāng)您苦苦思索一些邏輯問題的時(shí)候,這個(gè)功能尤為有效,您只需在感興趣的代碼前面插入trace函數(shù)即可。使用trace(true)語句表示跟蹤的開始,在您關(guān)注的代碼段末端插入trace(off)語句,禁用跟蹤功能。

您也可以通過打開控制臺(tái)窗口,輸入:
trace(true);
來使跟蹤選項(xiàng)可用。

下面是控制臺(tái)窗口中的一段輸出(沒有使用trace函數(shù))信息

            
              --------- Initializing MOD: Common ---------
Loading compiled script common/client/canvas.cs.
Loading compiled script common/client/audio.cs.
--------- Initializing MOD: Torque demo ---------
Loading compiled script demo/client/init.cs.
Loading compiled script demo/server/init.cs.
Loading compiled script demo/data/init.cs.
Loading compiled script demo/data/terrains/highplains/propertyMap.cs.

            
          

當(dāng)使用trace函數(shù)后,輸出信息就如下所示

            
              --------- Parsing Arguments ---------
Entering [demo]parseArgs()
Entering [common]parseArgs()
Leaving [common]parseArgs() - return
Leaving [demo]parseArgs() - return
Entering [demo]onStart()
Entering [common]onStart()
--------- Initializing MOD: Common ---------
Entering initCommon()
Loading compiled script common/client/canvas.cs.
Loading compiled script common/client/audio.cs.
Leaving initCommon() - return
Leaving [common]onStart() – return
--------- Initializing MOD: Torque demo ---------
Loading compiled script demo/client/init.cs.
Loading compiled script demo/server/init.cs.
Loading compiled script demo/data/init.cs.
Loading compiled script demo/data/terrains/highplains/propertyMap.cs.
Entering initServer()

            
          

VC中的TRACE宏:

  TRACE宏對(duì)于VC下程序調(diào)試來說是很有用的東西,有著類似printf的功能;該宏僅僅在程序的DEBUG版本中出現(xiàn),當(dāng)RELEASE的時(shí)候該宏就完全消失了,從而幫助你調(diào)式也在RELEASE的時(shí)候減少代碼量。

  使用非常簡(jiǎn)單,格式如下:

  TRACE("DDDDDDDDDDD");

  TRACE("wewe%d",333);

4. 在非MFC程序中使用調(diào)試宏ASSERT(),VERIFY()和 TRACE()

  • ASSERT()被測(cè)試它的參數(shù),若參數(shù)為0,則中斷執(zhí)行并打印一段說明消息。在 Release 版本的程序中它不起任何作用。
  • VERIFY()和 ASSERT()很相似,區(qū)別在于在 Release 版本中它仍然有效(譯者注:原作者在這里沒有講清楚,VERIFY()不會(huì)打印說明,只是會(huì)對(duì)參數(shù)表達(dá)式求值)。
  • ASSERT()使用的時(shí)候必須保證參數(shù)表達(dá)式中不能有函數(shù)調(diào)用(譯者注:ASSERT()宏在 Release 版本中不對(duì)表達(dá)式求值),因此對(duì)于任何有函數(shù)調(diào)用的參數(shù)表達(dá)式,應(yīng)該使用宏 VERIFY(),以保證表達(dá)式中的函數(shù)調(diào)用在 Release 版本中會(huì)被正確求值。
  • TRACE()基本上就是函數(shù) printf()的一個(gè)復(fù)制品,唯一的區(qū)別是它把結(jié)果輸出到調(diào)試窗口。在 Release 版本中,它也是無效的。
  • 這三個(gè)宏在 Release 版本中都不會(huì)產(chǎn)生任何實(shí)質(zhì)性的影響,它們是否起作用取決于是否定義了預(yù)定義了宏 _DEBUG。這是對(duì) Microsoft Visual C++ 而言,在其它的編譯器中可能其它不同的宏。
  •   Since it makes no sense to re-invent the wheel(譯者注:這好像是一句俗語,大致意思是“沒有必要(意義)自己從頭寫起”,但原句究竟如何,在下水平有限,實(shí)難猜出。故將原文放上,望高人賜教,感激不盡!),筆者在看了 MFC 的代碼之后類似地建立了自己的宏。對(duì)于 ASSERT()和 VERIFY()則去掉了花哨的“Debug assertion failed...”對(duì)話框,只是簡(jiǎn)單的產(chǎn)生一個(gè)單純的斷點(diǎn)中斷。

      要使用 ASSERT(),VERIFY()和 TRACE(),有兩個(gè)文件是必需的: debug.h debug.cpp 。首先需要在工程中的主要頭文件里中包含文件 debug.h 。因?yàn)樗旧頉]有包括其它任何頭文件,所以不必?fù)?dān)心會(huì)產(chǎn)生頭件的包含遞歸。另外還要將 debug.cpp 加入到工程中的源文件中。

    這里是代碼:

            
              
                
                  // file debug.h
                
                 #ifndef __DEBUG_H__ #define __DEBUG_H__ #ifdef _DEBUG void _trace(char *fmt, ...); #define ASSERT(x) {if(!(x)) _asm{int 0x03}} #define VERIFY(x) {if(!(x)) _asm{int 0x03}} 
                
                  // 譯注:為調(diào)試版本時(shí)產(chǎn)生中斷有效
                
                 #else #define ASSERT(x) #define VERIFY(x) x 
                
                  // 譯注:為發(fā)行版本時(shí)不產(chǎn)生中斷
                
                 #endif #ifdef _DEBUG #define TRACE _trace #else inline void _trace(LPCTSTR fmt, ...) { } #define TRACE 1 ? (void)0 : _trace #endif #endif 
                
                  // __DEBUG_H__
                
                
                  // file debug.cpp
                
                 #ifdef _DEBUG #include <stdio.h> #include <stdarg.h> #include <windows.h> void _trace(char *fmt, ...) { char out[1024]; va_list body; va_start(body, fmt); vsprintf(out, fmt, body); 
                
                  // 譯注:格式化輸入的字符串 fmtt
                
                 va_end(body); 
                
                  // 到輸出字符串 ou
                
                 OutputDebugString(out); 
                
                  // 譯注:輸出格式化后的字符串到調(diào)試器
                
                 } #endif
              
            
            
            
          

    譯者續(xù):一點(diǎn)小擴(kuò)展

      大家可以看到宏 TRACE()的最后,調(diào)用的是 OutPutDebugString()函數(shù),只能將信息輸出到調(diào)試器窗口中,但我們同樣也可以實(shí)現(xiàn) MFC 中的彈出式窗口,只要用 MessageBox()函數(shù)輸出就可以了。(不過……好像樣子也不一樣哎!)

    VC調(diào)試以及TRACE()函數(shù)的用法


    更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

    微信掃碼或搜索:z360901061

    微信掃一掃加我為好友

    QQ號(hào)聯(lián)系: 360901061

    您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

    【本文對(duì)您有幫助就好】

    您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺我的文章對(duì)您有幫助,請(qǐng)用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長(zhǎng)會(huì)非常 感謝您的哦!!!

    發(fā)表我的評(píng)論
    最新評(píng)論 總共0條評(píng)論
    主站蜘蛛池模板: 99久久精品国产国产毛片 | 国内自拍一二三四2021 | 婷婷在线成人免费观看搜索 | 亚洲精品视频在线播放 | 久久久久国产一级毛片高清板 | 激情视频日本 | 国内久久久久影院精品 | 久久天堂 | 欧美19p| 99热在线播放 | 九9热这里只有真品 | 久久国产一久久高清 | 一二三区无线码2021 | 寡妇一级a毛片免费播放 | 成人毛片大全 | 亚洲一级毛片欧美一级说乱 | 九九精品99| 色综合色狠狠天天综合色hd | 免费精品美女久久久久久久久久 | 91社区视频 | 久久久99精品免费观看精品 | 亚洲最大在线视频 | 亚洲欧洲视频在线观看 | 视频二区 中文字幕 欧美 | 大尺度视频网站久久久久久久久 | 国产高清美女一级毛片久久 | 国产乳摇福利视频在线观看 | 国产网站免费视频 | 水蜜桃网站| 99热视屏| 久久精品视频6 | 久久精品网址 | 中文字幕日韩一区二区 | 在线成人免费观看国产精品 | 亚洲免费人成在线视频观看 | 久久久久中文字幕 | 国产91久久最新观看地址 | 99国产精品热久久久久久夜夜嗨 | 成人在线免费小视频 | jzz欧美| 久久精品国产色蜜蜜麻豆 |