#define NDEBUGと#include<br>使用時に発生する問題.

2003 ワード

#includeと#define位置の相対関係は、プログラムの実行に影響を及ぼす可能性があります.
たとえば
#include
#define NDEBUG
#include
// #define NDEBUG
int main(){
    assert(0);
    std::cout << "hello world!" << std::endl;
    return 0;
}

運転後、出力hello world!、下段コード運転後
#include
#define NDEBUG
#include
// #define NDEBUG
int main(){
    assert(0);
    std::cout << "hello world!" << std::endl;
    return 0;
}

assertは情報を出力し、プログラムの実行を終了します.その原因は,#include#define NDEBUGがいずれも前処理中に展開され,cassertのソースコードが観察されるためである.
/*
 * assert.h
 * This file has no copyright assigned and is placed in the Public Domain.
 * This file is a part of the mingw-runtime package.
 * No warranty is given; refer to the file DISCLAIMER within the package.
 *
 * Define the assert macro for debug output.
 *
 */

/* We should be able to include this file multiple times to allow the assert
   macro to be enabled/disabled for different parts of code.  So don't add a
   header guard.  */

#ifndef RC_INVOKED

/* All the headers include this file. */
#include <_mingw.h>

#undef assert

#ifdef  __cplusplus
extern "C" {
#endif

#ifdef NDEBUG
/*
 * If not debugging, assert does nothing.
 */
#define assert(x)   ((void)0)

#else /* debugging enabled */

/*
 * CRTDLL nicely supplies a function which does the actual output and
 * call to abort.
 */
_CRTIMP void __cdecl __MINGW_NOTHROW _assert (const char*, const char*, int) __MINGW_ATTRIB_NORETURN;

/*
 * Definition of the assert macro.
 */
#define assert(e)       ((e) ? (void)0 : _assert(#e, __FILE__, __LINE__))

#endif  /* NDEBUG */

#ifdef  __cplusplus
}
#endif

#endif /* Not RC_INVOKED */


プリプロセッサは、実際にはヘッダファイルの内容を現在のファイル位置に挿入します.したがって、#define NDEBUGと#includeの相対的な位置は、プログラムの実行結果に影響します.コンパイルコマンドで-D NDEBUGを使用しても発生しません.