8

不感兴趣:

#define _ACD 5, 5, 5, 30

#define DEFAULT_NETWORK_TOKEN_KEY_CLASS   _ACD 

#define DEFAULT_NETWORK_TOKEN_KEY { DEFAULT_NETWORK_TOKEN_KEY_CLASS }

仅使用 DEFAULT_NETWORK_TOKEN_KEY_CLASS 宏,如何在 const unsigned char [] 中获取 _ACD 字符串化。

const uint8 startMsg[] = ?? DEFAULT_NETWORK_TOKEN_KEY_CLASS ;

只会产生_ACD

在此处获取 _ACD 的正确宏扩展是什么。在如何将具有数组的宏字符串化为#define a_macro {5,7,7,97} 的上下文中?

4

1 回答 1

11

(关于在没有充分理由的情况下不滥用 C 预处理器的标准免责声明适用于此。)

当然可以做你想做的事。你需要一个STRINGIFY宏和一些宏间接。

通常,STRINGIFY使用一级间接定义,以允许 C 预处理器在其参数进行字符串化之前扩展其参数。一种实现是:

/* The # operator converts symbol 'v' into a string */
#define STRINGIFY0(v) #v
#define STRINGIFY(v) STRINGIFY0(v)

但是,您会发现这还不够:

#define _ACD 5, 5, 5, 30
#define DEFAULT_NETWORK_TOKEN_KEY_CLASS   _ACD 
#define DEFAULT_NETWORK_TOKEN_KEY { DEFAULT_NETWORK_TOKEN_KEY_CLASS }

#define START_MSG STRINGIFY(DEFAULT_NETWORK_TOKEN_KEY_CLASS)
const char startMsg[] = START_MSG;

在这里,STRINGIFY(DEFAULT_NETWORK_TOKEN_KEY_CLASS)展开为STRINGIFY0(5,5,5,30),C 预处理器抱怨您提供STRINGIFY0了太多参数。

解决方案是延迟扩展,_ACD使其仅5,5,5,30在您想要的时候扩展。为此,请将其定义为类似函数的宏:

#define _ACD() 5, 5, 5, 30

这样,_ACD只有当你“调用”它时才会被扩展:_ACD(). DEFAULT_NETWORK_TOKEN_KEY_CLASS现在将扩展为_ACD,您必须通过“调用”它来进一步扩展它:DEFAULT_NETWORK_TOKEN_KEY_CLASS().

以下代码说明了解决方案:

#include <stdio.h>

#define STRINGIFY0(v) #v
#define STRINGIFY(v) STRINGIFY0(v)

#define _ACD() 5, 5, 5, 30
#define DEFAULT_NETWORK_TOKEN_KEY_CLASS   _ACD 
#define DEFAULT_NETWORK_TOKEN_KEY { DEFAULT_NETWORK_TOKEN_KEY_CLASS() }

#define START_MSG STRINGIFY(DEFAULT_NETWORK_TOKEN_KEY_CLASS)

const char startMsg[] = START_MSG;

int main(int argc, char** argv)
{
  printf("%s\n",startMsg);
  return 0;
}
于 2012-07-27T19:12:25.433 回答