c語言中的assert
⑴ c語言中assert斷言的用法
assert宏的原型定義在<assert.h>中,其作用是如果它的條件返回錯誤,則終止程序執行。
庫函數: assert.h
原型定義: void assert( int expression );
assert的作用是現計算表達式 expression ,如果其值為假(即為0),那麼它先向stderr列印一條出錯信息,然後通過調用 abort 來終止程序運行。常式:
#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>用法總結與注意事項:
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不能代替條件過濾
⑵ ASSERT在C語言中有什麼作用
所有的assert都是斷言,意思是程序運行到此處時,assert()內的表達式必須返回真,否則程序會拋出錯誤並停止運行。
在debug編譯模式時才產生代碼,在release編譯時,assert()會直接被忽略。
⑶ C語言用什麼語句終止整個程序
1、首先打開vc6.0,新建一個項目。
⑷ 在C語言中用\a表示鳴鈴是什麼意思啊
a是assert的縮寫, \a是轉義字元,可以使用printf一類的標准輸入輸出函數輸出鳴鈴。這個東西就是規定出來的,也說不出來為啥。會用就行了
⑸ C語言中用ASSERT調試的八大技巧
技巧1:記住ASSERT的定義
對許多開發人員來說,斷言是一個令人困惑的話題,因為它們的許多使用方式與其設計初衷背道而馳。我見到的最清晰的斷言定義是這樣的:
「斷言是在程序某個特定點的一個布爾表達式,除非程序中有缺陷(Bug),否則它的值將為真。」
想要理解上述斷言定義的開發人員應該留意下面三個要點:
·斷言會評估一個表達式是真還是假
·斷言是在代碼中的某個點對系統狀態的一種假設
·斷言會驗證系統假設,如果不為真,就表明代碼中有一個缺陷
技巧2:使用ASSERT驗證函數的先決條件
斷言非常適合契約式設計環境,在這種環境中,開發人員非常清晰地定義了某個函數的先決條件。斷言可以用來檢查該函數的輸入是否滿足先決條件。就拿圖1所示的代碼片段為例:
圖1:函數的先決條件
函數的STate輸入應該在定義的系統狀態范圍內。如果State不是有效的狀態值,那麼它就不是錯誤,而是缺陷!斷言可以用來驗證State是有效的假設,如圖2所示:
圖2:對函數先決條件應用斷言
在State不小於最大值的事件中,斷言表達式將被評估為假,程序於是將停止執行。停止程序執行可以讓開發人員很容易馬上看到哪裡的代碼出錯,而不是過段時間以後才知道。
技巧3:使用ASSERT驗證函數的後置條件
斷言也能用來驗證契約式設計環境中對某個函數輸出的假設。例如,如果前面定義的System_StateSet函數返回SystemState變數,開發人員可以預計它也在期望的范圍之內。斷言可以用來對缺陷進行監視,如圖3所示。
圖3:對函數後置條件應用斷言
開發人員在查看上述代碼後可能會感到這些檢查毫無意義。剛剛才設置好的SystemState怎麼就會出現大於SYSTEM_STATE_MAX的值呢?答案是這確實不應該出現,然而有時候會莫名其妙地發生改變,也許是通過中斷或並行線程,此時斷言可以立即標志出這個缺陷。
技巧4:不要把ASSERT用於錯誤處理
在記住斷言定義之後,開發人員應該切記:斷言是用於檢測缺陷的,不能用於錯誤處理。錯誤處理是設計用於響應錯誤的用戶輸入和意外的事件順序的軟體。錯誤在系統中預料是會發生的,但僅僅是因為有無效的輸入而並不意味著代碼中有缺陷。錯誤處理應該與缺陷尋找分開來。錯誤使用斷言的一個典型例子是,在試圖打開一個文件用於讀取時去檢查文件的指針,如圖4所示。
圖4:ASSERT的不當使用
讀者可以清楚地看到,試圖打開文件的結果與文件系統的狀態和用戶數據有關,而與代碼中的缺陷一點關系也沒有。開發人員應該編寫錯誤處理程序,而不是用斷言,以便在文件不存在時,錯誤處理程序可以用一些默認可用數據來創建它,以便後續代碼繼續操作。
技巧5:ASSERT僅對開發有意義,不能用於生產
開發ASSERT宏的原始意圖是在開發過程中啟用它,在後面生產時要禁用。可以用NDEBUG宏激活和禁用ASSERT。正確實施的斷言在被禁用後應該對嵌入式系統基本沒有影響。
問題是,如果測試是在斷言啟用的情況下進行的(為了捕捉任何缺陷,應該這樣做),那麼現在禁用斷言將導致交付的產品與測試的產品處於不同的狀態。斷言確實會佔用一些代碼空間,但更重要的是,它們需要佔用少量的時鍾周期來評估它們的布爾表達式。禁用ASSERT可能對具有有限資源的裸機系統的執行時序產生很大影響,從而導致在生產系統中產生新的缺陷。開發團隊需要判斷是否值得冒關閉斷言的.風險。
一種替代方案是保留斷言在激活狀態,而將它們的輸出重定向到一個系統日誌。這樣可以確保任何揮之不去的缺陷很容易被識別,而且能避免中止系統的運行,而中止系統可不是明智之舉。
技巧6:不允許斷言有副作用
ASSERT的默認實現允許開發人員包含一段可執行代碼作為布爾表達式的一部分。舉例來說,一個狀態變數可以被實現為表達式的一部分並傳遞給ASSERT。但如果傳遞給ASSERT的表達式有副作用,也就是說,它會改變嵌入式系統的狀態,那麼禁用斷言將改變系統的行為。開發人員應該確保他們的表達式沒有副作用,否則他們需要冒險在系統中增加只針對產品代碼喚醒的休眠時間缺陷。
技巧7:斷言應該占代碼的1%至3%
每個開發人員對於代碼庫(Code Base)中應該有多少個斷言都有自己的主見。大家一致同意的一個數字是,代碼庫中的斷言佔比應該大於0。斷言為開發人員提供了一種在代碼庫中發生缺陷的時刻發現它的好方法。調試是在開發嵌入式系統中最浪費時間並令人沮喪的事情之一。不管開發人員認可的佔比是1%、3%還是5%,使用斷言肯定對你有利,並會使開發嵌入式軟體變得多少有些趣味。
技巧8:將斷言用作可執行代碼注釋
斷言可以生成極好的注釋!編寫出色的表達式可以確切地告訴開發人員在代碼的某個給定點應該預料發生什麼事情。開發人員應該做好他們斷言的架構,幫助人們更清楚地理解系統中發生的事情,進而幫助減少缺陷。
小結
斷言是一種出色的工具,但有太多的嵌入式軟體開發人員忽視了這一工具。本文討論的八個技巧只是如何正確使用斷言的冰山一角。接下來讀者就可以在測試平台中建立和開始使用斷言,並研究它們在實際的嵌入式系統中是如何工作的。
⑹ 關於C語言中assert等語法在嵌入式應用當中的含義
assert_param(IS_GPIO_MODE(GPIO_InitStruct->GPIO_Mode));
意思是:IS_GPIO_MODE(GPIO_InitStruct->GPIO_Mode)這個判斷條件必須為真,否則程序就會進入死循環。
一般assert用來判斷必須為真的一些條件,防止程序出現意外錯誤。
例如:
assert(汽車有4個輪子)//這個是必須成立的條件
開汽車
⑺ c語言中什麼情況需要檢查函數參數的有效性
一般來說是對外提供的介面一定要檢查參數的有效性;模塊內部使用的介面要求稍低,但為了多人合作和方便後人接手,最好也加上參數有效性檢查,既可以檢查參數有效性,又可以指明參數應有的特徵,方便讀代碼。
另外在正常的參數有效性檢查之外,還可以使用assert斷言檢查,用作調試,但是不能簡單的用assert來代替參數檢查。