CPP:避免宏功能参数的宏扩展

con*_*roy 9 c c++ macros stringification c-preprocessor

我想做的(用于记录目的)是这样的:

编写此代码是为了显示我的问题,实际代码很复杂,是的,我有充分的理由在C++上使用宏=)

# define LIB_SOME 1
# define LIB_OTHER 2

# define WHERE "at file #a, line #l, function #f: "
// (look for syntax hightlighting error at SO xd)
# define LOG_ERROR_SIMPLE(ptr, lib, str) ptr->log ("ERROR " str \
                                                   " at library " #lib);
# define LOG_ERROR(ptr, lib, str) LOG_ERROR_SIMPLE(ptr, lib, WHERE str)

LOG_ERROR_SIMPLE (this, LIB_SOME, "doing something")
LOG_ERROR (this, LIB_OTHER, "doing something else")
Run Code Online (Sandbox Code Playgroud)

LOG_ERROR_SIMPLE() 写入lib参数的字符串化(由""包围的宏名称)

但随后LOG_ERROR写入已扩展的宏的字符串化("2").这是预期的,因为lib在扩展和调用之前得到了扩展LOG_ERROR_SIMPLE.但这不是我需要的.

基本上我的问题是:如何在调用另一个宏函数时避免宏函数参数的宏扩展?

我使用了一个避免宏扩展的技巧:

  LOG_ERROR(ptr, lib, str, x) LOG_ERROR_SIMPLE(ptr, x##lib, WHERE str)

  LOG_ERROR(this, LIB_OTHER, "some error",)
Run Code Online (Sandbox Code Playgroud)

(粘贴x和lib产生LIB_OTHER并且此值用于调用LOG_ERROR_SIMPLE,在调用之前不扩展宏)

有一些方法可以在不使用技巧的情况下获得相同的行为?

Gre*_*osz 8

我正在做:

#include <cstdio>

#define FOO 1
#define BAR 2

#define LOG_SIMPLE(ptr, lib, str) printf("%s\n", #lib);
#define LOG(ptr, lib, str) LOG_SIMPLE(ptr, ##lib, str)

int main()
{
  LOG_SIMPLE(0, FOO, "some error");
  LOG(0, BAR, "some other error");
}
Run Code Online (Sandbox Code Playgroud)

打印出:

FOO
BAR
Run Code Online (Sandbox Code Playgroud)

适用于MSVC2005 但不适用于gcc/g ++.


编辑:要使它与gcc/g ++一起使用,你可以滥用可变参数宏:

#include <stdio.h>

#define FOO 1
#define BAR 2

#define LOG_SIMPLE(ptr, str, lib) printf("%s\n", #lib);
#define LOG(ptr, str, lib, ...) LOG_SIMPLE(ptr, str, lib##__VA_ARGS__)

int main()
{
  LOG_SIMPLE(0, "some error", FOO);
  LOG(0, "some other error", BAR);
  LOG(0, "some other error", FOO, BAR);
}
Run Code Online (Sandbox Code Playgroud)

但是,不要使用带有太多参数的宏,这是你的纪律.打印出MSVC2005

FOO
BAR
FOO2
Run Code Online (Sandbox Code Playgroud)

而gcc打印出来

FOO
BAR
FOOBAR
Run Code Online (Sandbox Code Playgroud)

  • 格雷戈里,我印象深刻,但我不能再投票给你了:) (2认同)