GCC中常见预定义宏的使用

来源:互联网 发布:淘宝店铺怎么没有收藏 编辑:程序博客网 时间:2024/04/28 10:28

在标准C以及各中编译器中定义了一些对象宏, 这些宏的名称以"__"开头和结尾, 并且都是大写字符. 这些预定义宏可以被#undef, 也可以被重定义。

      在ANSI C标准中定义了__FILE__,__LINE__,__DATA__,__TIME__,__STDC__等标准的预定义宏。GCC对其进行扩展,也定义了多个预定义宏。

      概括起来GCC中可使用的预定义宏涵盖了如下几方面的信息:
      1、宿主的信息:GNU的版本,编译器的版本,类型的相关信息,字节序信息等。
      2、编译动作的信息:编译的日期、时间;编译时是否进行了时间或空间上的优化;定义的inline是否被编译器执行等。
      3、文件的信息:文件名称、函数名称、行数信息、文件最后修改时间等等。
      4、计数信息:__COUNTER__,__INCLUDE_LEVEL__等。

      下面是一些常见的预定义宏的使用方法。

1、__FILE__,__LINE__,FUNCTION__
      这是最常用到的预定义宏的组合,表示文件名、行数和函数名,用于程序运行期异常的跟踪。如:

 

 

      运行./tset时:
      line 9 in main.c(main) : two argvs are needed
      Aborted

     

      运行./test kkk时:
      line 12 in func.c(func) : file not exist
      Aborted

      运行./test test时成功。

      可见通过使用__FILE__,__LINE__,FUNCTION__宏,可以帮助我们精确的定位出现异常的文件、函数和行数。

2、__BASE_FILE__

      这个宏是和__FILE__相对应的,表示主输入文件的名字,对于源文件而言__FILE__和__BASE_FILE__是一样的;对于头文件二者才可能不同。比如在上个例子中,__LINE__这个宏是在myassert.h文件中定义的,被main.c和func.c包含之后__FILE__的值
分别变成了main.c和func.c。但是当我们希望知道MyAssert这个宏具体实在哪个文件(实际上是myassert.h)中定义的话,就需要用到__BASE_FILE__。
     下面的例子可以帮助加深理解:

 

  

 

       gcc main.c &&./a.out 得到:

       basefile.h
       main.c

 3、__DATE__,__TIME__
       用于得到最后一次编译的日期和时间(字符串形式):

 

 

       gcc main.c &&./a.out 得到:
       DATE : Jan 27 2011
       TIME : 17:12:55

4、__TIMESTAMP__
       和__TIME__的格式相同。同于得到本文件最后一次被修改的时间。

5、__GNUC__、__GNUC_MINOR__、__GNUC_MINOR__、__GNUC_PATCHLEVEL__
       用于得到GNU版本:

 

 

6、__VERSION__
     用于得到编译器的版本      

 

 

      gcc main.c && ./a.out得到:

      Version : 4.1.2 (Gentoo 4.1.2 p1.0.2)
      可以和gcc -v相互验证

7、__COUNTER__
      自身计数器,用于记录以前编译过程中出现的__COUNTER__的次数,从0开始计数。常用于构造一系列的变量名称,函数名称等。如:

 

 

      gcc main.c &&a.out得到结果:
      0
      1
      这里使用__COUNTER__构造了两个变量:var0,var1。

8、__INCLUDE_LEVEL__
      用于表示文件被包含的计数,从0开始递增,常作为递归包含的限制条件。如:

 

 

      gcc main.c && ./a.out,得到结果:
      1 2 3 4 5 6 7 8 9
      在这个例子中文件rep.h自包含了9次,执行了9次REP(BLAH)。

      实际上,__INCLUDE_LEVEL__最多的是和#include __FILE__组合使用,用于表示一个递归。如:

 

 

      gcc main.c && ./a.out得到结果:
      1 2 3 4 5 6 7 8 9 10 9 8 7 6 5 4 3 2 1

参考:

http://gcc.gnu.org/onlinedocs/cpp/Standard-Predefined-Macros.html#Standard-Predefined-Macros
http://gcc.gnu.org/onlinedocs/cpp/Common-Predefined-Macros.html#Common-Predefined-Macros

 

原创粉丝点击