assert宏的原型定義在<assert.h>中,其作用是如果它的條件返回錯誤,則終止程序執行,原型定義如下:
#include <assert.h>
void assert( int expression );
assert的作用是現計算表達式 expression ,如果其值爲假(即爲0),那麼它先向stderr打印一條出錯信息,然後通過調用 abort 來終止程序運行。
請看下面的程序清單badptr.c:
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
int main( void )
{
FILE *fp;
fp = fopen( "test.txt", "w" ); //以可寫的方式打開一個文件,如果不存在就創建一個同名文件
assert( fp ); //所以這裏不會出錯
fclose( fp );
fp = fopen( "noexitfile.txt", "r" ); //以只讀的方式打開一個文件,如果不存在就打開文件失敗
assert( fp ); //所以這裏可能會出錯
fclose( fp ); //如果出錯,程序永遠都執行不到這裏來
return 0;
}
在命令行下運行該程序,結果如下:
[root@localhost error_process]# gcc badptr.c
[root@localhost error_process]# ./a.out
a.out: badptr.c:14: main: Assertion `fp'' failed.
已放棄
使用assert的缺點是,頻繁的調用會極大的影響程序的性能,增加額外的開銷。
在調試結束後,可以通過在包含#include <assert.h>的語句之前插入 #define NDEBUG 來禁用assert調用,示例代碼如下:
#include <stdio.h>
#define NDEBUG
#include <assert.h>
assert的用法總結與注意事項:
1、在函數開始處檢驗傳入參數的合法性,如:
int resetBufferSize(int nNewSize)
{
//函數功能:改變緩衝區大小,
//函數參數:nNewSize 緩衝區新長度
//函數返回值:緩衝區當前長度
//說明:保持原信息內容不變
//nNewSize<=0表示清除緩衝區
assert(nNewSize >= 0);
assert(nNewSize <= MAX_BUFFER_SIZE);
...
}
2、每個assert只檢驗一個條件,因爲同時檢驗多個條件時,如果斷言失敗,無法直觀地判斷是哪個條件失敗,比較下面兩種形式的代碼:
不好: assert(nOffset>=0 && nOffset+nSize<=m_nInfomationSize);
好: assert(nOffset >= 0);
assert(nOffset+nSize <= m_nInfomationSize);
3、不能使用改變環境的語句,因爲assert只在DEBUG時生效,這麼做的話會使程序在真正運行時遇到問題
錯誤,例如:
assert(i++ < 100);
如果在執行之前i=100,那麼這條語句就不會執行,i++這條命令就沒有執行。
正確的寫法: assert(i < 100);
i++;
4、assert和後面的語句應空一行,以形成邏輯和視覺上的一致感。
5)注意,有的地方,assert不能代替條件過濾。
文章出處:http://www.diybl.com/course/3_program/c++/cppjs/20071111/85534.html
以下內容摘自《高質量C/C++編程指南》Page 41-42...
程序一般分爲Debug 版本和Release 版本,Debug 版本用於內部調試,Release 版本發行給用戶使用。assert 是僅在Debug 版本起作用的宏,它用於檢查“不應該”發生的情況
assert 不是函數,而是宏。程序員可以把assert看成一個在任何系統狀態下都可以安全使用的無害測試手段。如果程序在 assert 處終止了,並不是說含有該assert 的函數有錯誤,而是調用者出了差錯,assert 可以幫助我們找到發生錯誤的原因。
assert宏中應該包含的元素:判斷條件;輸出當前斷言失敗的位置(文件、行數等);返回錯誤;終止程序...
幾種典型的assert的寫法:
VC中的寫法:
#define ASSERT(f) /
do /
{ /
if (!(f) && AfxAssertFailedLine(THIS_FILE, __LINE__)) /
AfxDebugBreak(); /
} while (0) /
#define _ASSERT(expr) /
do { if (!(expr) && /
(1 == _CrtDbgReport(_CRT_ASSERT, __FILE__, __LINE__, NULL, NULL))) /
_CrtDbgBreak(); } while (0)
其他平臺的寫法:
# define ASSERT(x) ((x) || (dbg_printf("assertion failed ("__FILE__":%d): /"%s/"/n",__LINE__,#x), break_point(), FALSE))