C++ 用可变参数包装一个包装器宏?

标签 c++ macros variadic-functions

问题

我有一个方法可以通过用宏替换函数来包装函数,这样我就可以记录调用和返回代码。这是一个有效的示例:

int rc;
int foo(int a, int b);
int bar(int a, char *b, int *c);

void    LogRet(char *fn, char *file, char *from, int ln, int ret)
{
    printf("%s.%s.%d: %s()  ret:%08x\n", file, from, ln, fn, ret);
}   

#define foo(args, ...)  (rc = (foo)(args, ##__VA_ARGS__), LogRet("foo", __FILE__, __FUNCTION__, __LINE__, rc), rc)

#define bar(args, ...)  (rc = (bar)(args, ##__VA_ARGS__), LogRet("bar", __FILE__, __FUNCTION__, __LINE__, rc), rc)

它所替代的函数的宏会调用函数并记录函数名称、调用位置以及返回代码。包装任何函数使用相同的语法,只需要在宏中替换函数名称 3 次。我想做的是创建包装器宏,其中为 foo 重新定义的宏类似于:

#define foo(args, ...) WRAPPPER(foo)

我了解 stringify 和 double stringify 的基础知识,但我什至无法让 WRAPPER 宏执行真正的函数调用。理想情况下,我想将其简化为单个 WRAP(foo) 语句。原因是,我有大约 100 个或更多的函数我想包装,它想通过一个简单的强制包含文件来简单地完成它。我得出的结论是这是不可能的,但我想在放弃这个想法之前先在这里问一下。我正在使用 clang 和 vc++,如果这有什么不同的话,但在我调试很多不同的系统时,在任何编译器上使用它会很好。


改编乔纳森·莱弗勒的回答

因为我是新来的,所以我不确定这应该是一个单独的答案还是编辑更新。这基本上是 Jonathan Leffler 的回答。这是一个功能示例。虽然它调用的 2 个函数毫无意义,但目标是拥有一个可以用任何参数列表包装任何函数的宏。我的主要用途是在有问题的大型代码库中记录使用流程。此外,我的原始样本有一个明显的弱点。通过将返回代码存储在全局中,如果没有在 TLS 中进行繁琐的准备,它就不是线程安全的。现在删除了全局变量,宏不再使用序列运算符返回值,它由日志函数保留和返回。此外,正如 Augurar 在 Jonathan 的例子中指出和展示的那样。如果宏与函数声明在同一文件中声明,则需要括号。

#include <stdio.h>
#include <string.h>

int foo(int a, int b);
int bar(int a, char *b, int *c);


#if defined (_DEBUG)  || defined (DEBUG)
// Short version of __FILE__ without path requires runtime parsing
#define __SFILE__ (strrchr(__FILE__, '\\') ? strrchr(__FILE__, '\\') + 1 : __FILE__)
#ifdef WIN32
  #define WRAPPER(func, ...) LogRet(#func, __SFILE__, __FUNCTION__, __LINE__, (func)(__VA_ARGS__))
#else
  #define WRAPPER(func, ...) LogRet(#func, __SFILE__, __func__, __LINE__, (func)(__VA_ARGS__))
#endif

inline  int LogRet(const char *fn, const char *file, const char *from, int ln, int ret)
{
    printf("%s.%s.%d: %s()  ret:%08x\n", file, from, ln, fn, ret);
    return ret;
}   

#define foo(...) WRAPPER(foo, __VA_ARGS__)
#define bar(...) WRAPPER(bar, __VA_ARGS__)

#endif


int main(void)
{
    int x = foo(1, 2);
    bar(2, "doubled", &x);
    
    return 0;
}   

#ifdef foo
  #undef foo
  #undef bar
#endif

// If and only if the function definition is in the same file with the macros, you must either undefine the macros or
// parenthesize the function  e.g.   int (foo)(int a, int b) { ... }

int foo(int a, int b)   
{
    printf("%d + %d = %d\n", a, b, a + b);
    return a + b;
}   

int (bar)(int a, char *b, int *c)
{
    printf("%d  %s = %d\n", *c, b, a * *c);
    return *c * a;
}

发布构建输出:

1 + 2 = 3
3  doubled = 6

调试构建输出:

1 + 2 = 3
test.cpp.main.35: foo()  ret:00000003
3  doubled = 6
test.cpp.main.36: bar()  ret:00000006

主要好处是不必在代码中查找每次出现的 foo() 或 bar() 即可插入调试打印以记录调用和结果或您要插入的任何调试代码。

最佳答案

这段代码看起来好像符合您的要求:

#include <stdio.h>

int rc;
int foo(int a, int b);
int bar(int a, char *b, int *c);

extern void LogRet(const char *fn, const char *file, const char *from, int ln, int ret);

void LogRet(const char *fn, const char *file, const char *from, int ln, int ret)
{
    printf("%s.%s.%d: %s()  ret:%08x\n", file, from, ln, fn, ret);
}

#define foo(args, ...)  (rc = (foo)(args, ##__VA_ARGS__), LogRet("foo", __FILE__, __FUNCTION__, __LINE__, rc), rc)

#define bar(args, ...)  (rc = (bar)(args, ##__VA_ARGS__), LogRet("bar", __FILE__, __FUNCTION__, __LINE__, rc), rc)

extern void caller1(void);

void caller1(void)
{
    int d;
    int e = foo(1, 2);
    int f = bar(3, "abc", &d);
    printf("%s(): %d + %d + %d = %d\n", __func__, d, e, f, d + e + f);
}

#undef foo
#undef bar

#define WRAPPER(func, ...) ((rc = (func)(__VA_ARGS__)), LogRet(#func, __FILE__, __func__, __LINE__, rc), rc)

#define foo(...) WRAPPER(foo, __VA_ARGS__)
#define bar(...) WRAPPER(bar, __VA_ARGS__)

extern void caller2(void);

void caller2(void)
{
    int d;
    int e = foo(2, 3);
    int f = bar(3, "abc", &d);
    printf("%s(): %d + %d + %d = %d\n", __func__, d, e, f, d + e + f);
}

int (foo)(int a, int b)
{
    return (a + b) % 3;
}

int (bar)(int a, char *b, int *c)
{
    int d = b[a];
    *c = d;
    return a + d;
}

int main(void)
{
    caller1();
    caller2();
    return 0;
}

示例输出:

wrapper.c.caller1.23: foo()  ret:00000000
wrapper.c.caller1.24: bar()  ret:00000003
caller1(): 0 + 0 + 3 = 3
wrapper.c.caller2.41: foo()  ret:00000002
wrapper.c.caller2.42: bar()  ret:00000003
caller2(): 0 + 2 + 3 = 5

预处理后的源代码(不包括 #include <stdio.h> 的输出):

# 2 "wrapper.c" 2

int rc;
int foo(int a, int b);
int bar(int a, char *b, int *c);

extern void LogRet(const char *fn, const char *file, const char *from, int ln, int ret);

void LogRet(const char *fn, const char *file, const char *from, int ln, int ret)
{
    printf("%s.%s.%d: %s()  ret:%08x\n", file, from, ln, fn, ret);
}





extern void caller1(void);

void caller1(void)
{
    int d;
    int e = (rc = (foo)(1, 2), LogRet("foo", "wrapper.c", __FUNCTION__, 23, rc), rc);
    int f = (rc = (bar)(3, "abc", &d), LogRet("bar", "wrapper.c", __FUNCTION__, 24, rc), rc);
    printf("%s(): %d + %d + %d = %d\n", __func__, d, e, f, d + e + f);
}
# 36 "wrapper.c"
extern void caller2(void);

void caller2(void)
{
    int d;
    int e = ((rc = (foo)(2, 3)), LogRet("foo", "wrapper.c", __func__, 41, rc), rc);
    int f = ((rc = (bar)(3, "abc", &d)), LogRet("bar", "wrapper.c", __func__, 42, rc), rc);
    printf("%s(): %d + %d + %d = %d\n", __func__, d, e, f, d + e + f);
}

int (foo)(int a, int b)
{
    return (a + b) % 3;
}

int (bar)(int a, char *b, int *c)
{
    int d = b[a];
    *c = d;
    return a + d;
}

int main(void)
{
    caller1();
    caller2();
    return 0;
}

在 Mac OS X 10.9.2 上使用 GCC 4.8.2 测试。

关于C++ 用可变参数包装一个包装器宏?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22395738/

相关文章:

c++ - 如何用宏删除括号?

c++ - 编译错误: "stddef.h: No such file or directory"

c++ - 在二叉树解决方案中删除

单独文件中的 C++ 内部模板类

c++ - 它是如何工作的,如何使用宏编译 TCL 的 C++ 扩展而没有主要功能

c - 使用 gcc 编译一个显示 "undefined reference to ` 中止的项目'”

c++ - 如何使用标准 API 打开和读取具有 unicode 路径或文件名的文件的内容?

c++ - 修改 TestAssert.h (cppunit) - 为什么包含的顺序对宏扩展很重要?

python - **(双星/星号)和*(星/星号)对参数有何作用?

c++ - 可变参数列表是否以 null 结尾?