How do you perform macro expansion within #ifdef?

白昼怎懂夜的黑 提交于 2020-01-03 11:29:32

问题


I have some fairly generic code which uses preprocessor macros to add a certain prefix onto other macros. This is a much simplified example of what happens:

#define MY_VAR(x) prefix_##x

"prefix_" is actually defined elsewhere, so it will be different each time the file is included. It works well, but now I have some code I would like to skip if one of the tokens doesn't exist, but this doesn't work:

#if defined MY_VAR(hello)

What I want it to expand to is this:

#ifdef prefix_hello

But I can't figure out how. I need to use the MY_VAR() macro to do the expansion, so I can't just hardcode the name. (It's actually for some testing code, the same code gets included with a different prefix each time to test a bunch of classes, and I want to skip a couple of tests for a handful of the classes.)

Is this possible with the C++ preprocessor?

Update:

Here is some semi-compilable code to demonstrate the problem further: (to avoid squishing it into the comments below)

#define PREFIX hello

#define DO_COMBINE(p, x)  p ## _ ## x
#define COMBINE(p, x)     DO_COMBINE(p, x)
#define MY_VAR(x)         COMBINE(PREFIX, x)

// MY_VAR(test) should evaluate to hello_test

#define hello_test "blah blah"

// This doesn't work
#ifdef MY_VAR(test)
  printf("%s\n", MY_VAR(test));
#endif

回答1:


Is there more to your program than this question describes? The directive

#define MY_VAR(x) prefix_##x

defines exactly one preprocessor identifier. The call

blah ^&* blah MY_VAR(hello) bleh <>? bleh

simply goes in one end of the preprocessor and comes out the other without defining anything.

Without some other magic happening, you can't expect the preprocessor to remember what arguments have been passed into what macros over the course of the preceding source code.

You can do something like

#define prefix_test 1
#if MY_VAR(test) == 1
#undef prefix_test // optional, "be clean"
...
#endif
#undef prefix_test

to query whether the prefix currently has the particular value prefix_. (Undefined identifiers are replaced with zero.)




回答2:


I think you've got to the level where the preprocessor won't cut it any more; it's really quite simple-minded. Have you considered using templates instead? (Assuming that they're meaningful for your problem, of course.)




回答3:


This is in response to your updated question. For each possible hello_test, after any possibility of #define hello_test "blah blah", add the lines:

#ifndef hello_test
#define hello_test (char*)0
#endif

Then change your test to:

if (MY_VAR(test))
    printf("%s\n", MY_VAR(test));

Or, as an alternative, before all the #define ..."blah blah"'s, add the line(s):

static const char * const MY_VAR(test);

for all possible values of MY_VAR and test. This will avoid hard-coding "hello_test". (The second const here removes the gcc warning: ‘hello_test’ defined but not used.)



来源:https://stackoverflow.com/questions/2703516/how-do-you-perform-macro-expansion-within-ifdef

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!