Turbo C 程序的調(diào)試
一個程序設計好了以后, 通常會有一些錯誤, 查找和修改程序中的錯誤是令人頭痛的事。Turbo C集成開發(fā)環(huán)境提供了一調(diào)試裝置, 使得這一個工作容易了許多, 程序調(diào)試達到了編譯和運行級。
一、TC消息窗口
使用TC最好的理由之一是它允許用戶修改語法錯誤(編譯時) 和評估編譯器給出的警告。TC將編譯器和連接器發(fā)出的消息收集到一緩沖區(qū)中, 然后在消息窗口中顯示, 這樣在訪問源代碼的同時, 還能一下看到這些消息。
現(xiàn)將上面的HELLO.C制造一點語法錯誤, 將第一行包含語句的#去掉, 再去掉第五行printf語句中的后引號。現(xiàn)在程序看上去是這樣的:
include
main()
{
printf("Hello, world\n);
}
按CTRL+F9重新編譯之。編譯窗口將顯示有多少錯誤和警告: 應為兩個錯誤, 0個警告。
當看見編譯窗口中的Press anykey提示時, 按空格鍵, 消息窗口立刻被激活,亮條出現(xiàn)在第一個錯誤或警告上, 這時編輯窗口中也會有一亮條--- 它標志著編譯器給出的錯誤或警告在源代碼中的相應位置。
這時可用光標鍵將消息窗口中的亮條上下移動, 注意到編輯窗口中的亮條也隨著跟蹤源代碼中錯誤發(fā)生的位置。如果將亮條置于"compile"上, 則編輯器顯示文件的最后位置。
如果消息窗口太長看不見, 可用左、右光標水平滾動消息, 為了一次能夠多看點信息, 可按F5放大消息窗口。放大后, 編輯窗口不可見了, 因此不進行錯誤跟蹤,F(xiàn)在, 保持分屏模式。
為了改正錯誤, 將消息窗口中的亮條置于第一個錯誤消息上, 回車, 光標移到編輯窗口中錯誤產(chǎn)生處, 注意, 編輯器狀態(tài)給出所選消息( 這在放大模式下是有用的)改正之。(將第一行拿走的#重新寫上)。
當不只一個錯誤時, 可用兩種方法來修改下一錯誤。
第一種方法和前面一樣, 按F6回到消息窗口選擇想修改的下一條消息。
第二種方法不用回到消息窗口, 只要按Alt+F8, 編譯器就會將光標移至消息窗口中列的下一個錯誤。按Alt+F7可移至前一個錯誤。
這兩種方法各有長短, 視情況而定。有時源代碼中一個愚蠢的錯誤把編譯弄糊涂了, 產(chǎn)生好多消息, 這時選擇修改第一條消息就使得其余的一些錯誤消息沒有什么意義了, 這種情況發(fā)生時, 使用方法一會方便些, 一修改完第一個錯誤之后回到消息窗口, 再滾動到下一個有意義的消息上, 選擇之。在別的情況下, 按Alt+F8會方便得多。
記住, Alt+F7和Alt+F8是熱鍵, TC中無論何時均起作用。因此在消息窗口中按Alt+F8得到的不是當前亮行消息, 而是下一個消息(按Enter選擇當前消息)。但如果沒別的編譯消息, Alt+F8就不起作用了。
注: 可以如此法選擇連接消息, 但它們不跟蹤源文件。在修改語法錯誤的過程當中, 經(jīng)常需要增加、刪除正文, 編輯器是記住的, 依然能正確定位錯誤位置。沒有必要記住行號和增加、刪除的正文行。
二、Turbo C集成調(diào)試器
一旦修改好語法錯誤之后程序編譯就沒什么問題了, 但還是可能不按要求運行, 因為可能有邏輯錯誤(運行錯誤)。這種錯誤跟蹤就無助于發(fā)現(xiàn)錯誤位置了。TC有一個集成調(diào)試器可以跟蹤運行錯誤。通過調(diào)試器可以運行, 在斷點處暫停,檢查變量的值, 甚至可以改變之, 以看程序會有什么反應。
Turbo C集成調(diào)試器是源程序級的調(diào)試器, 即用同你編寫程序一樣的" 語言"來控制調(diào)試器。例如, 為了顯示數(shù)組中的一個元素的值, 可告訴調(diào)試器顯示這樣的表達式的值:
Ctrl+F4 Debug/Eavluate 計算表達式, 允許修改變量的值。
Debug/Find Function 查找函數(shù)定義, 顯示在編輯窗口中。 僅
在調(diào)試時才有效。
Ctrl+F3 Debug/Call Stack 顯示調(diào)用棧, 可顯示任何函數(shù)的當前執(zhí)
行位置, 其方法是在調(diào)用棧中選擇相應的
函數(shù)名。僅在調(diào)試時有效。
Debug/Source Debugging 控制是否允許調(diào)試: 置為On時, 集成調(diào)
試器和單獨調(diào)試器均可用 ; 置為
Standalone時, 只能用單獨調(diào)試器
調(diào)試, 雖然還能在TC中運行; 置為
None時, 在.EXE文件不置調(diào)試信息,
兩種調(diào)試均不能調(diào)試。
Ctrl+F4 Break/Watch/Add Watch 增加一監(jiān)視表達式。
Break/Watch/Delete Watch 刪除一監(jiān)視表達式。
Break/Watch/Edit 編輯一監(jiān)視表達式。
Break/Watch/Remove All 刪除所有監(jiān)視表達式。
Watches
Ctrl+F8 Break/Watch/Toggle 設置或清除光標所在行的斷點。
Breakpoint
Break/Watch/Clear 刪除程序中所有斷點。
Breakpoint
Break/Watch/Next 顯示下一斷點
Breakpoint
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
表3. 調(diào)試器菜單命令及其熱鍵
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
熱鍵 菜單命令 功能
───────────────────────────────────
F5 在整屏和分屏之間放大縮小活動窗口。
Alt+F5 將顯示轉(zhuǎn)到用戶屏, 擊任意鍵返回。
F6 在編輯窗口與監(jiān)視窗口或消息窗口間
切換。
Alt+F6 若編輯窗口是活動的, 轉(zhuǎn)到最近一次
裝入編輯器的文件; 若下面窗口是活
動的, 則在監(jiān)視窗口和消息窗口間切換。
Ctrl+F9 Run/Run 調(diào)試運行或不調(diào)試運行程序, 必要時
編譯、連接源文件, 若編譯、 連接時
Debug/Source Debuging和O/C/C/OBJ
Debug Information為On, 則運行程序
到斷點或程序末尾。
Project/Remove Messages 刪除消息窗口中的內(nèi)容。
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
判斷程序是否出錯或者為什么出錯是編程中最有挑戰(zhàn)意義的一方面。這里建義你進行預防性設計, 具體做法為:
(1). 將代碼寫清楚點, 應作適當縮進, 使用文字說明和描述性的變量名。
(2). 代碼應簡單, 把精力放在簡單語句中的復雜表達式, 而不是一些復雜語句。Turbo C的代碼優(yōu)化機制將大大提高代碼的效率, 而且調(diào)試、閱讀、 修改起來容易。
(3). 盡量用目的簡單、好定義的函數(shù)構(gòu)建程序。 這會便于編制調(diào)試程序和分析結(jié)果, 而且閱讀、修改起來也要容易一些。
(4). 應盡量減少各個函數(shù)要求的數(shù)據(jù)和改變數(shù)據(jù)的元素個數(shù)。這也會便于編制測試程序和分析結(jié)果; 同樣便于閱讀、修改程序。并且還限制了出錯函數(shù)可能造成的巨大混亂的牽涉面, 便得能在一個調(diào)試節(jié)中多運行函數(shù)幾次。
(5). 要留心編寫程序中的公共函數(shù), 或者說在其它程序中可再用的函數(shù)。編寫、調(diào)試一個一般性的函數(shù)通常要比編寫、調(diào)試兩個或更多的特殊函數(shù)要容易。
Turbo C編譯、連接和運行時的常見錯誤
一、編譯時的常見錯誤
1. 數(shù)據(jù)類型錯誤。此類錯誤是初學者編程時的常見現(xiàn)象, 下面是一些要引起注意的錯誤:
(1) 所有變量和常量必須要加以說明。
(2) 變量只能賦給相同類型的數(shù)據(jù)。
(3) 對scanf()語句, 用戶可能輸入錯誤類型的數(shù)據(jù)項, 這將導致運行時出錯, 并報出錯信息。為避免這樣的錯誤出現(xiàn), 你就提示用戶輸入正確類型的數(shù)據(jù)。
(4) 在執(zhí)行算術(shù)運算時要注意:
a. 根據(jù)語法規(guī)則書寫雙精度數(shù)字。要寫0.5, 而不是寫.5; 要寫1.0, 而不是1。盡管C語言會自動地把整型轉(zhuǎn)換成雙精度型, 但書寫雙精度型是個好習慣。讓C語言為你做強行轉(zhuǎn)換這是一種效率不高的程序設計風格。 這有可能導致轉(zhuǎn)換產(chǎn)生錯誤。
b. 不要用0除。這是一個災難性的錯誤, 它會導致程序失敗, 不管C語言的什么版本, 都是如此, 執(zhí)行除法運算要特別小心。
c. 確保所有的雙精度數(shù)(包括那些程序輸入用的雙精度數(shù)) 是在實數(shù)范圍之內(nèi)。
d. 所有整數(shù)必須在整數(shù)允許的范圍內(nèi)。這適用于所有計算結(jié)果, 包括中間結(jié)果。
2. 將函數(shù)后面的";"忘掉。此時錯誤提示色棒將停在該語句下的一行, 并顯示:
Statement missing ; in function <函數(shù)名>
3. 給宏指令如#include, #define等語句尾加了";"號。
4. "{"和"}"、"("和")"、"/*"和"*/"不匹配。 引時色棒將位于錯誤所在的行, 并提示出有關(guān)丟掉括號的信息。
5. 沒有用#include指令說明頭文件, 錯誤信息提示有關(guān)該函數(shù)所使用的參數(shù)未定義。
6. 使用了Turbo C保留關(guān)鍵字作為標識符, 此時將提示定義了太多數(shù)據(jù)類型。
7. 將定義變量語句放在了執(zhí)行語句后面。此時會提示語法錯誤。
8. 使用了未定義的變量, 此時屏幕顯示:
Undefined symbol '<變量名>' in function <函數(shù)名>
9. 警告錯誤太多。忽略這些警告錯誤并不影響程序的執(zhí)行和結(jié)果。編譯時當警告錯誤數(shù)目大于某一規(guī)定值時(缺省為100)便退出編譯器, 這時應改變集成開發(fā)環(huán)境Options/Compiler/Errors中的有關(guān)警告錯誤檢查開關(guān)為off。
10. 將關(guān)系符"=="誤用作賦值號"="。此時屏幕顯示:
Lvalue required in function <函數(shù)名>
二、連接時的常見錯誤
1. 將Turbo C庫函數(shù)名寫錯。這種情況下在連接時將會認為此函數(shù)是用戶自定義函數(shù)。此時屏幕顯示:
Undefined symbol '<函數(shù)名>' in <程序名>
2. 多個文件連接時, 沒有在"Project/Project name中指定項目文件 (.PRJ 文件), 此時出現(xiàn)找不到函數(shù)的錯誤。
3. 子函數(shù)在說明和定義時類型不一致。
4. 程序調(diào)用的子函數(shù)沒有定義。
三、運行時的常見錯誤
1. 路徑名錯誤。在MS-DOS中, 斜杠(\)表示一個目錄名; 而在Turbo C 中斜杠是個某個字符串的一個轉(zhuǎn)義字符, 這樣, 在用Turbo C 字符串給出一個路徑名時應考慮"\"的轉(zhuǎn)義的作用。例如, 有這樣一條語句:
file=fopen("c:\new\tbc.dat", "rb");
目的是打開C盤中NEW目錄中的TBC.DAT文件, 但做不到。這里"\"后面緊接的分別是"n"及"t", "\n"及"\t"將被分別編譯為換行及tab字符, DOS將認為它是不正確的文件名而拒絕接受, 因為文件名中不能和換行或tab字符。正確的寫法應為:
file=fopen("c:\\new\\tbc.dat", "rb");
2. 格式化輸入輸出時, 規(guī)定的類型與變量本身的類型不一致。例如:
float l;
printf("%c", l);
3. scanf()函數(shù)中將變量地址寫成變量。例如:
int l;
scanf("%d", l);
4. 循環(huán)語句中, 循環(huán)控制變量在每次循環(huán)未進行修改, 使循環(huán)成為無限循環(huán)。
5. switch語句中沒有使用break語句。
6. 將賦值號"="誤用作關(guān)系符"=="。
7. 多層條件語句的if和else不配對。
8. 用動態(tài)內(nèi)存分配函數(shù)malloc()或calloc()分配的內(nèi)存區(qū)使用完之后, 未用free()函數(shù)釋放, 會導致函數(shù)前幾次調(diào)用正常, 而后面調(diào)用時發(fā)生死機現(xiàn)象,不能返回操作系統(tǒng)。其原因是因為沒用空間可供分配, 而占用了操作系統(tǒng)在內(nèi)存中的某些空間。
9. 使用了動態(tài)分配內(nèi)存不成功的指針, 造成系統(tǒng)破壞。
10. 在對文件操作時, 沒有在使用完及時關(guān)閉打開的文件。
相關(guān)推薦:計算機等考二級C語言備考:C語言/C++編譯過程北京 | 天津 | 上海 | 江蘇 | 山東 |
安徽 | 浙江 | 江西 | 福建 | 深圳 |
廣東 | 河北 | 湖南 | 廣西 | 河南 |
海南 | 湖北 | 四川 | 重慶 | 云南 |
貴州 | 西藏 | 新疆 | 陜西 | 山西 |
寧夏 | 甘肅 | 青海 | 遼寧 | 吉林 |
黑龍江 | 內(nèi)蒙古 |