以下は目次です。
なんらかのヘッダファイルをインクルードしなくても使える、あらかじめ定義されたマクロのことを、事前定義マクロ (predefined macro) といいます。事前定義マクロは多数ありますが、代表的なものを取り上げると以下があります。
__FILE__は、このマクロを使用したソースファイルの名前を表す文字列リテラルに置換されます。
// main.c
#include <stdio.h>
int main(void)
{
(__FILE__);
puts}
実行結果:
main.c
__FILE__ によって置換された結果は文字列リテラルですから、puts関数に渡せばそのまま出力できます。
なお、ファイル名を強制的に変更する #line指令という命令があり、その影響を受けます。これは後で説明します。
__LINE__は、このマクロを使用したファイル上の行の番号を表す整数型の定数に置換されます。
#include <stdio.h>
int main(void)
{
("%d\n", __LINE__);
printf}
実行結果:
5
なお、行番号を強制的に変更する #line指令という命令があり、その影響を受けます。これは後で説明します。
__DATE__は、コンパイルを行ったときの日付に置換されます。置換された結果は文字列リテラルです。
#include <stdio.h>
int main(void)
{
(__DATE__);
puts}
実行結果:
Mar 11 2023
置換結果は、日付の形式は「月 日 年」の順番であり、間を半角スペースで空けたものです。「月」の部分は asctime関数 と同じ表現になります(asctime関数は、第51章で取り上げます)。「日」の部分は必ず2桁分の幅を取り、10未満のときは上位桁が空白文字になります[1]。
__TIME__は、コンパイルを行ったときの時刻に置換されます。置換された結果は文字列です。
#include <stdio.h>
int main(void)
{
(__TIME__);
puts}
実行結果:
13:28:16
時刻の形式は「時:分:秒」の順番で、:
で区切られています。24時間形式で表現され、それぞれ 2桁の
10進数です。
__STDC__は、処理系が標準規格に準拠していれば 1 に置換されます。準拠していない場合には、そもそもこのマクロが定義されていません。0 などの値に置換されるわけではないので、使い方に注意してください。
#include <stdio.h>
int main(void)
{
#ifdef __STDC__
("Conforms to standards.");
puts#else
("Not standard compliant.");
puts#endif
}
実行結果(Visual Studio):
Not standard compliant.
実行結果(gcc):
Conforms to standards.
Visual Studio と gcc とで実行結果が異なっています。Visual Studio での結果は意外かもしれません。
Visual Studio では、プロジェクトのプロパティを開き、「構成プロパティ」→「C/C++」→「言語」を選択し、「言語拡張を無効にする」を「はい(/Za)」に変更すれば、__STDC__ が定義されます。
__STDC_VERSION__ は、処理系がどの標準規格に対応しているかを表す、long int型の定数に置き換わります。
以下の表のようになっています。
対応規格 | 結果 |
---|---|
C99 | 199901L |
C95 | 199409L |
C89以前、または規格に非対応 | このマクロが定義されていない |
【C11】__STDC_VERSION__ の置換結果は 201112L に更新されました。
#include <stdio.h>
int main(void)
{
#ifdef __STDC_VERSION__
("%ld\n", __STDC_VERSION__);
printf#else
("__STDC_VERSION__ undefined.\n");
printf#endif
}
実行結果(Visual Studio):
__STDC_VERSION__ undefined.
実行結果(gcc 6.4.0 (MinGW-w64)):
201112
Visual Studio では、__STDC__ の定義の有無とも関係なく、つねに __STDC_VERSION__ は定義されないようです。gcc では、__STDC_VERSION__ が定義されているので、値が出力されます。
Visual Studio ではできませんが、本来的には、__STDC__ の結果と、__STDC_VERSION__ の結果とを組み合わせて、対応規格を知ることができます。
__STDC__ | __STDC_VERSION__ | 意味 |
---|---|---|
定義されている | 199901L | C99規格に対応 |
定義されている | 199409L | C95規格に対応 |
定義されている | 定義されていない | C89規格に対応 |
定義されていない | — | C言語の標準規格に合致していない |
__STDC_HOSTED__ は、処理系が、ホスト処理系であれば 1 に、フリースタンディング処理系であれば 0 に置換されます。
ホスト処理系 (hosted implementation) とは、プログラムの実行が OS の支援によって行われる環境のことです。このような環境では、C言語の標準規格にあるすべての機能が使用できます。
フリースタンディング処理系 (freestanding implementation) とは、OS の支援無しでプログラムが実行される環境で、C言語の標準規格にある機能の一部しか使用できません。
#include <stdio.h>
int main(void)
{
("%d\n", __STDC_HOSTED__);
printf}
実行結果 :
1
事前定義マクロと非常によく似ているものに、事前定義識別子 (predefined identifier) があります。名前のとおりですが、事前定義マクロと違ってマクロではなく、識別子です。いわば、変数を暗黙的に宣言しているのと似ており、スコープの概念があります。
C99 で定義されている事前定義識別子は __func__ だけです。
__func__は、関数の本体でのみ使用でき、その関数名の文字列リテラルを得られます。より具体的には、関数の冒頭部分に、以下のコードがあるかのようにコンパイルされます。
static const char __func__[] = "関数名";
使用例は以下のようになります。
#include <stdio.h>
void func(void);
int main(void)
{
(__func__);
puts
();
func}
void func(void)
{
(__func__);
puts}
実行結果:
main
func
#line は、ソースファイルの行番号や名前を制御します。
#line の使い方は次の2通りあります。
#line 行番号
#line 行番号 ソースファイル名
いずれにしても、#line
を記述した行の行番号を「行番号」に指定した数とみなすようにプリプロセッサに指示を与えます。「ソースファイル名」を指定した場合は、同様に
#line
を記述した行以降のソースファイル名を「ソースファイル名」とみなすようになります。その結果、__FILE__
や __LINE__
の置換結果が影響を受けます。また、処理系がエラーや警告などを報告するときに表示されるソースファイル名や行番号にも影響することがあります。
#include <stdio.h>
int main(void)
{
(__FILE__);
puts("%d\n", __LINE__);
printf
#line 1000 "new_name.c"
(__FILE__);
puts("%d\n", __LINE__);
printf}
実行結果:
c:\test_program\main.c
6
new_name.c
1001
最後にプラグマ (pragma) を紹介します。プラグマは、処理系が独自に実装している、標準でない機能を使うための構文で、次のように #pragma を使って記述します。
#pragma *****
*****
の部分は、使おうとする機能によって異なります。標準のものではないため、処理系のドキュメントを確認するなどして理解しなければなりません。
【上級】解説は省きますが、#pramga を #define の「置換後の文字の並び」のところで使えない問題を解決する方法として、_Pragma演算子があります[2]。
比較的よく認知されているプラグマとして、#pragma
once があります。これはインクルードガード(第24章)を簡単に実現します。ヘッダファイルの先頭に
#pragma once
と記述するだけで、インクルードガードの役割を果たします。
#pragma once
// ほかのコード
主要なコンパイラではインクルードガードとして機能しますが、あくまでも処理系定義の機能です。これを認識しないコンパイラでは、単にこの記述を無視するかもしれません。
問題① プログラムの実行直後に、そのソースファイルをコンパイルしたときの日付・時刻を出力してみてください。
問題② ソースコード上のある地点で、ソースファイルの名前と行数を出力できるデバッグ関数を次のように作成しました。しかし、この関数には実用上の問題があります。問題点を指摘してください。
void print_log(void)
{
("File: %s Line: %d\n", __FILE__, __LINE__);
printf}
問題③ 自分の使っているコンパイラに用意されている pragma指令について調べてください。
≪さらに古い更新履歴を展開する≫
Programming Place Plus のトップページへ
はてなブックマーク に保存 | Pocket に保存 | Facebook でシェア |
X で ポスト/フォロー | LINE で送る | noteで書く |
RSS | 管理者情報 | プライバシーポリシー |