变长参数的 Tracer

    技术2022-05-11  188

    几天前,在论坛看到这么一则讨论:在宏定义中怎么使用可变参数?(http://expert.csdn.net/Expert/topic/2925/2925165.xml)。楼主希望能定义这样的macro:

    #define fun1(a, b, ...)   fun2(__FILE__, __LINE__, a, b, ...)

    我猜楼主是想写trace,如果不能使用可变参数的macro,那么就得像MFC那样写一堆TRACE macros:

    // 取自 MFC 7.1 的 afx.h// The following trace macros are provided for backward compatiblity//  (they also take a fixed number of parameters which provides//   some amount of extra error checking)#define TRACE0(sz)               TRACE(_T("%s"), _T(sz))#define TRACE1(sz, p1)           TRACE(_T(sz), p1)#define TRACE2(sz, p1, p2)       TRACE(_T(sz), p1, p2)#define TRACE3(sz, p1, p2, p3)   TRACE(_T(sz), p1, p2, p3)

    太丑陋了!还好,C99标准支持Variadic Macros,在GCC中,可以这么写:

    // http://gcc.gnu.org/onlinedocs/gcc/Variadic-Macros.html#define debug(format, ...)  fprintf(stderr, format, __VA_ARGS__)

    还可以顺便打印文件名和行号:

    #define debug(format, ...)  do {/                fprintf(stderr, "%s (%d): ", __FILE__, __LINE__);/                fprintf(stderr, format, __VA_ARGS__);/                } while (0)

    但可惜Visual C++ 7.1还不支持这项功能:( 不过我们在C++中至少可以绕弯解决,做到既能自动记录文件名和行号,又能使用变长参数调用。这个办法不是我独创的,实际上ATL的atltrace.h中就有它的实现(CtraceFileAndLineInfo class),我在Code Project也找到了相同的实现(http://www.codeproject.com/debug/location_trace.asp),甚至在CUJ的C++ Experts Forum 也能看到相近的做法(http://www.cuj.com/documents/s=8250/cujcexp2106alexandr/),当然Alexandrescu的办法技巧性更强。

    思路:写一个重载了 operator() 的class,令 TRACE 宏返回该class的一个object:

    #include #include

    #ifndef NDEBUG  // debug mode

    class tracer{public:  tracer(const char* file, int line)    : file_(file), line_(line)  {}    void operator()(const char* fmt, ...)  {    va_list ap;        // print the file name and line number    fprintf(stderr, "%s (%d): ", file_, line_);        va_start(ap, fmt);    vfprintf(stderr, fmt, ap);    va_end(ap);        fprintf(stderr, "/r/n"); // print the new-line character  }

    private:  // copy-ctor and operator=  tracer(const tracer&);  tracer& operator=(const tracer&);

    private:  const char* file_;  int         line_;};#define TRACE (tracer(__FILE__, __LINE__))#else  // NDEBUG#define TRACE (void)#endif // NDEBUG

    int main(){#ifndef NDEBUG  tracer(__FILE__, __LINE__)("%x", 123);#endif

      TRACE("%s", "Happy debugging.");}

    这样做是multithreading-safe的。G++ 3.3.1 / Visual C++ 7.1 / Borland C++ 5.5.1 通过。


    最新回复(0)