C++ 中assert()函数用法总结
assert宏的原型定義在<assert.h>中,其作用是如果它的條件返回錯誤,則終止程序執(zhí)行,原型定義:
| 1 2 | #include <assert.h> void assert( int expression ); |
assert的作用是現(xiàn)計算表達(dá)式 expression ,如果其值為假(即為0),那么它先向stderr打印一條出錯信息,然后通過調(diào)用 abort 來終止程序運(yùn)行。
請看下面的程序清單badptr.c:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | #include <stdio.h> #include <assert.h> #include <stdlib.h> int main( void ) { ????FILE *fp; ???? ????fp = fopen( "test.txt", "w" );//以可寫的方式打開一個文件,如果不存在就創(chuàng)建一個同名文件 ????assert( fp );????????????? //所以這里不會出錯 ????fclose( fp ); ???? ????fp = fopen( "noexitfile.txt", "r" );//以只讀的方式打開一個文件,如果不存在就打開文件失敗 ????assert( fp );????????????? //所以這里出錯 ????fclose( fp );????????????? //程序永遠(yuǎn)都執(zhí)行不到這里來 ????return 0; } |
| 1 2 3 | [root@localhost error_process]# gcc badptr.c [root@localhost error_process]# ./a.out a.out: badptr.c:14: main: Assertion `fp' failed. |
使用assert()的缺點(diǎn)是,頻繁的調(diào)用會極大的影響程序的性能,增加額外的開銷。在調(diào)試結(jié)束后,可以通過在包含#include <assert.h>的語句之前插入 #define NDEBUG 來禁用assert調(diào)用,
示例代碼如下:
| 1 2 3 | #include <stdio.h> #define NDEBUG #include <assert.h> |
用法總結(jié)與注意事項(xiàng):
1)在函數(shù)開始處檢驗(yàn)傳入?yún)?shù)的合法性如:
| 1 2 3 4 5 6 7 8 9 10 | int resetBufferSize(int nNewSize) { //功能:改變緩沖區(qū)大小, //參數(shù):nNewSize 緩沖區(qū)新長度 //返回值:緩沖區(qū)當(dāng)前長度? //說明:保持原信息內(nèi)容不變?? nNewSize<=0表示清除緩沖區(qū) assert(nNewSize >= 0); assert(nNewSize <= MAX_BUFFER_SIZE); ... } |
2)每個assert只檢驗(yàn)一個條件,因?yàn)橥瑫r檢驗(yàn)多個條件時,如果斷言失敗,無法直觀的判斷是哪個條件失敗,如:
不好:
| 1 | assert(nOffset>=0 && nOffset+nSize<=m_nInfomationSize); |
好:
| 1 2 | assert(nOffset >= 0); assert(nOffset+nSize <= m_nInfomationSize); |
3)不能使用改變環(huán)境的語句,因?yàn)閍ssert只在DEBUG個生效,如果這么做,會使用程序在真正運(yùn)行時遇到問題,如:
錯誤:
| 1 | assert(i++ < 100); |
這是因?yàn)槿绻鲥e,比如在執(zhí)行之前i=100,那么這條語句就不會執(zhí)行,那么i++這條命令就沒有執(zhí)行。
正確:
| 1 2 | assert(i < 100); i++; |
4)assert和后面的語句應(yīng)空一行,以形成邏輯和視覺上的一致感。
5)有的地方,assert不能代替條件過濾。
以下是使用斷言的幾個原則:
(1)使用斷言捕捉不應(yīng)該發(fā)生的非法情況。不要混淆非法情況與錯誤情況之間的區(qū)別,后者是必然存在的并且是一定要作出處理的。
(2)使用斷言對函數(shù)的參數(shù)進(jìn)行確認(rèn)。
(3)在編寫函數(shù)時,要進(jìn)行反復(fù)的考查,并且自問:“我打算做哪些假定?”一旦確定了的假定,就要使用斷言對假定進(jìn)行檢查。
(4)一般教科書都鼓勵程序員們進(jìn)行防錯性的程序設(shè)計,但要記住這種編程風(fēng)格會隱瞞錯誤。當(dāng)進(jìn)行防錯性編程時,如果“不可能發(fā)生”的事情的確發(fā)生了,則要使用斷言進(jìn)行報警。??
ASSERT ()是一個調(diào)試程序時經(jīng)常使用的宏,在程序運(yùn)行時它計算括號內(nèi)的表達(dá)式,如果表達(dá)式為FALSE (0), 程序?qū)蟾驽e誤,并終止執(zhí)行。如果表達(dá)式不為0,則繼續(xù)執(zhí)行后面的語句。這個宏通常原來判斷程序中是否出現(xiàn)了明顯非法的數(shù)據(jù),如果出現(xiàn)了終止程序以免導(dǎo)致嚴(yán)重后果,同時也便于查找錯誤。?
ASSERT只有在Debug版本中才有效,如果編譯為Release版本則被忽略。?
比較好的在程序中使用assert的地方:
(1)空指針檢查。例如,針對一個函數(shù)的參數(shù)進(jìn)行空指針檢查。你可以這樣使用:assert (pointer != NULL);,產(chǎn)生的錯誤會像這樣:Assertion ‘pointer != ((void *)0)' failed。這樣,當(dāng)出現(xiàn)空指針時,你的程序就會退出,并很好的給出錯誤信息。
(2)檢查函數(shù)參數(shù)的值。例如,如果一個函數(shù)只能在它的一個參數(shù)foo為正值的時候被調(diào)用,你可以在函數(shù)開始時這樣寫:assert (foo > 0);,這將幫助你檢測函數(shù)的錯誤使用,這也給源代碼閱讀者很清晰的印象,那就是在這里對函數(shù)的參數(shù)值有限制。
?感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
總結(jié)
以上是生活随笔為你收集整理的C++ 中assert()函数用法总结的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 局部静态变量Static详解
- 下一篇: mysql通过拷贝数据文件的方式进行数据