#include <stdio.h> #define A ".5.6" #define SAMPLE_RATE_CONFIG 5 #define STR1(R) #R #define STR3(R) STR1(R) A #define STR4 STR3(SAMPLE_RATE_CONFIG) int main() { printf("same rate config = %s ", STR4 ); return 0; }
结果如下
预编译之后如下 int main() { printf("same rate config = %s ", "5" ".5.6" ); return 0; } 结果如下: same rate config = 5.5.6
适用场景举例
流媒体处理系统中,采样率是个很重要的参数,很多地方都要用,如果用来分配内存,则
#define SAMPLE_RATE 16000
如果用来配置第三方库,则
#define SAMPLE_RATE "16000"
这样每次改采样率都要改两处地方,容易遗漏
网上搜到第一篇文章,匆忙试了试,并未达到我想要的效果,然后在GNU的官网找到了第二篇文章,看懂了,原来第一篇文章的技巧(也是翻译第二篇文章的)仅适用于宏函数的场景
还好天无绝人之路,第二篇文章末尾3段,讲述了实现我想要效果的方法,就是二级stringfication
#include <stdio.h>
#define SAMPLE_RATE 16000
#define STR1(R) #R
#define STR2(R) STR1(R)
int main()
{
printf("sample rate = " STR2(SAMPLE_RATE) "
");
return 0;
}
运行输出
sample rate = 16000
注意:
二级stringfication会将宏定义原样输出,即,如果16000加上括弧(有经验的程序员都懂),则STR2输出的也是带括弧的字符串,这通常不是你想要的
可以这样绕过:
给SAMPLE_RATE定义一个别名,并给别名加上括弧,在需要整型时用别名,在需要字符串时用原名
#include <stdio.h>
#define SAMPLE_RATE_CONFIG 16000
#define SAMPLE_RATE (SAMPLE_RATE_CONFIG)
#define STR1(R) #R
#define STR2(R) STR1(R)
int main()
{
printf("sample rate = %d
", SAMPLE_RATE);
printf("sample rate config = %s
", STR2(SAMPLE_RATE_CONFIG));
return 0;
}
运行输出
sample rate = 16000
sample rate config = 16000
拼接
最近在编写一个项目的代码时,需要在宏定义中连接多个字符串,具体来说就是,先定义一个软件版本号,然后再定义一个硬件版本号, 然后再将他们拼合起来生成一个综合版本号。这些动作我都希望在宏定义中直接完成,提供代码的可读性和可移植性。
类似于下面这样的:
#define SOFTWARE_VERSION "Software:V1.00"
#define HARDWARE_VERSION "Hardware:V1.00"
#define SYSTEM_VERSION ????
但是,经过实际测试,以上的代码,只能用于KEIL/ADS/IAR等集成编译环境中。如果是在linux下,使用gcc编译器的话,上述代码就会出错,目前尚未查出具体原因。经过一番折腾后,发现gcc环境下,如果要连接多个字符串,直接使用空格连接就行了。所以将其改为如下语句就可以了:
#define SOFTWARE_VERSION "Software:V1.00"
#define HARDWARE_VERSION "Hardware:V1.00"
#define SYSTEM_VERSION SOFTWARE_VERSION HARDWARE_VERSION
keil
#define SOFTWARE_VERSION "Software:V1.00"
#define HARDWARE_VERSION "Hardware:V1.00"
#define SYSTEM_VERSION SOFTWARE_VERSION##" "##HARDWARE_VERSION