1. 程式人生 > >斷言(ASSERT)的用法

斷言(ASSERT)的用法

 它居然是個巨集,並且作用並非“報錯”。

  在經過對其進行一定了解之後,對其作用及用法有了一定的瞭解,assert()的用法像是一種“契約式程式設計”,在我的理解中,其表達的意思就是,程式在我的假設條件下,能夠正常良好的運作,其實就相當於一個if語句:

if(假設成立)
{
     程式正常執行;
}
else
{
      報錯&&終止程式!(避免由程式執行引起更大的錯誤)  
}

  但是這樣寫的話,就會有無數個if語句,甚至會出現,一個if語句的括號從檔案頭到檔案尾,並且大多數情況下,我們要進行驗證的假設,只是屬於偶然性事件,又或者我們僅僅想測試一下,一些最壞情況是否發生,所以這裡有了assert().

   assert巨集的原型定義在assert.h中,其作用是如果它的條件返回錯誤,則終止程式執行. 

1 #include "assert.h" 
2 void assert( int expression );

  assert的作用是現計算表示式 expression ,如果其值為假(即為0),那麼它先向stderr(stderr -- 標準錯誤輸出裝置)列印一條出錯資訊,然後通過呼叫 abort(異常終止一個程序,致所有的流被關閉和沖洗) 來終止程式執行。
 使用assert的缺點是,頻繁的呼叫會極大的影響程式的效能,增加額外的開銷。 
  在除錯結束後,可以通過在包含#include 的語句之前插入 #define NDEBUG 來禁用assert呼叫,示例程式碼如下: 

1 #include 
2 #define NDEBUG 
3 #include

用法總結與注意事項: 

  1)在函式開始處檢驗傳入引數的合法性 
  如: 

 1 int resetBufferSize(int nNewSize) 
 2 { 
 3 //功能:改變緩衝區大小, 
 4 //引數:nNewSize 緩衝區新長度 
 5 //返回值:緩衝區當前長度 
 6 //說明:保持原資訊內容不變 nNewSize<=0表示清除緩衝區  7 assert(nNewSize >= 0); 
 8 assert(nNewSize <= MAX_BUFFER_SIZE); 
 9  
10 ... 
11
}

  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不能代替條件過濾 

  程式一般分為Debug 版本和Release 版本,Debug 版本用於內部除錯,Release 版本發行給使用者使用。斷言assert 是僅在Debug 版本起作用的巨集,它用於檢查“不應該”發生的情況。以下是一個記憶體複製程式,在執行過程中,如果assert 的引數為假,那麼程式就會中止(一般地還會出現提示對話,說明在什麼地方引發了assert)。 

以下是使用斷言的幾個原則: 

  (1)使用斷言捕捉不應該發生的非法情況。不要混淆非法情況與錯誤情況之間的區別,後者是必然存在的並且是一定要作出處理的。 

  (2)使用斷言對函式的引數進行確認。 

  (3)在編寫函式時,要進行反覆的考查,並且自問:“我打算做哪些假定?”一旦確定了的假定,就要使用斷言對假定進行檢查。 

  (4)一般教科書都鼓勵程式設計師們進行防錯性的程式設計,但要記住這種程式設計風格會隱瞞錯誤。當進行防錯性程式設計時,如果“不可能發生”的事情的確發生了,則要使用斷言進行報警。 

  ASSERT ()是一個除錯程式時經常使用的巨集,在程式執行時它計算括號內的表示式,如果表示式為FALSE (0), 程式將報告錯誤,並終止執行。如果表示式不為0,則繼續執行後面的語句。這個巨集通常原來判斷程式中是否出現了明顯非法的資料,如果出現了終止程式以免導致嚴重後果,同時也便於查詢錯誤。  

ASSERT只有在Debug版本中才有效,如果編譯為Release版本則被忽略。