在编译时提取宏常量中的位数

Edv*_*olm 3 c c-preprocessor

我需要做一些预处理器魔术.假设我有一个全局常量

#define MAX_VALUE 99999
Run Code Online (Sandbox Code Playgroud)

我需要做的是在编译时以十进制表示形式提取此常量的长度.换句话说,我不想要另一个常数

#define MAX_VALUE_STRLEN 5
Run Code Online (Sandbox Code Playgroud)

污染全局命名空间,我不想在修改MAX_VALUE的情况下在代码中添加需要更改的位置.如果我有一个数字文字,那么我可以做类似的事情

#define INTLEN(x) (sizeof(#x)/sizeof((#x)[0]) - 1)
Run Code Online (Sandbox Code Playgroud)

然后在编译时INTLEN(99999)扩展到5.不幸的是,我做不了类似的事情

INTLEN(MAX_VALUE),
Run Code Online (Sandbox Code Playgroud)

因为预处理器首先扩展INTLEN,所以我得到了

 (sizeof("MAX_VALUE")/sizeof(("MAX_VALUE")[0]) - 1)
Run Code Online (Sandbox Code Playgroud)

是否有预处理器技巧可以达到我想要的效果?我应该能够安全地忽略的另一个棘手的问题是,如果有人决定添加一个类型注释,例如,99999L到常量,我仍然可以获得正确的值,这可以做得足够通用吗?

The*_*ant 7

Stringify使用#和两级宏扩展,然后切断终止NUL:

#define MAX_VALUE 99999

#define STRINGIFY(x) #x

#define LENGTH(x) (sizeof(STRINGIFY(x)) - 1)

#include <stdio.h>    

int main()
{
    size_t n = LENGTH(MAX_VALUE);
    printf("length = %zu\n", n);
    return 0;
}
Run Code Online (Sandbox Code Playgroud)