gcc中调试代码常用的宏

来源:互联网 发布:淘宝女童模特施欣怡 编辑:程序博客网 时间:2024/06/14 00:09

     红色是比较常用的意义__FILE__      本文件名__LINE__本行位于该文件的第几行__FUNCTION__函数名__PRETTY_FUNCTION__c语言中和同上,C++中稍有区别

__VA_ARGS__

格式化输出

__VA_ARGS__的使用例子:

#define ct_verbose(level, chfr, ...)  ct_log("", 0, "", level , NULL, chfr, __VA_ARGS__)

其他的更简单,就不写了。

  在标准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__宏,可以帮助我们精确的定位出现异常的文件、函数和行数。

 

下面是glib把调试信息定义成宏:

 #if defined(__GNUC__) && (__GNUC__ < 3) && !defined(__cplusplus)
#  define G_STRLOC __FILE__ ":" G_STRINGIFY (__LINE__) ":" __PRETTY_FUNCTION__ "()"
#else
#  define G_STRLOC __FILE__ ":" G_STRINGIFY (__LINE__)
#endif

 

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


原创粉丝点击