如何在C中串联两个字符串宏?

时间:2019-06-24 11:13:26

标签: c macros concatenation preprocessor

我正在尝试为我的程序实现VERSION宏,在某些情况下需要更改该宏。

宏版本是通过Makefile定义的(git info放在此处),它是一个字符串。 现在,我有一组#define开关,并且我希望VERSION反映其中的哪个开关。现在看起来如下(main.h):

#define COMPLEX_DEPOSITION // This is switch. later in code it is used in #ifdef...#endif construction.

#ifdef COMPLEX_DEPOSITION
#define CD "_COMP_DEP" // this is the string I want to put in the end of VERSION
#define VERSION_ VERSION CD

#undef VERSION // this is to suppress 'macro redefinition' warning
#define VERSION VERSION_
#undef VERSION_
#endif

好吧,我遇到了很多错误,其中大多数使我认为C预处理程序可以按随机顺序处理文件中的行:(

后来我有一个更复杂的东西,目的是使VERSION -> VERSION_WLT_GAP_2

#define WIRESLIFETIMES

#ifdef WIRESLIFETIMES
#define GAP 2
#define VERSION_ (VERSION ## "_WLT_GAP_" ## #GAP)
#define VERSION VERSION_
#undef VERSION_
#endif

我不知道该怎么办,如果可能的话

2 个答案:

答案 0 :(得分:3)

字符串文字在彼此相邻放置时会自然连接

"foo" "bar""foobar"相同。

对于第二个示例,您可能想要:

#define CAT_(A,B) A##B
#define CAT(A,B) CAT_(A,B)

#define GAP 2
#define VERSION CAT(VERSION_WLT_GAP_ , GAP)

VERSION //expands to VERSION_WLT_GAP_2

我建议稍微玩gcc -E / clang -E,以了解宏的工作原理, 在尝试与他们构成任何复杂事物之前。

答案 1 :(得分:1)

好吧,答案似乎如下:

// https://stackoverflow.com/questions/5256313/c-c-macro-string-concatenation
// Concatenate preprocessor tokens A and B without expanding macro definitions (however, if invoked from a macro, macro arguments are expanded).
#define PPCAT_NX(A, B) A ## B

// Concatenate preprocessor tokens A and B after macro-expanding them.
#define PPCAT(A, B) PPCAT_NX(A, B)

// Turn A into a string literal without expanding macro definitions (however, if invoked from a macro, macro arguments are expanded).
#define STRINGIZE_NX(A) #A

// Turn A into a string literal after macro-expanding it.
#define STR(A) STRINGIZE_NX(A)


#define COMPLEX_DEPOSITION

#ifdef COMPLEX_DEPOSITION
#define CD "_COMPDEP"
#else
#define CD ""
#endif


#define WIRESLIFETIMES

#ifdef WIRESLIFETIMES
#define GAP 2
#define WLT STR(PPCAT(_WLT:G, GAP))
#define DISABLE_METROPOLIS
#else
#define WLT ""
#endif

#define VERSION VERSIONX CD WLT

产生V008.1-11-g68a9c89cb4-dirty_COMPDEP_WLT:G2,对此我感到满意。

必须注意,我在Makefile中将-DVERSION=...更改为-DVERSIONX=...