在工程初期,某个回调函数还没有被实现,但是希望在能够打印出来
举例1:
在项目初期,单元测试有多个用例不想直接开发,但希望先定义出来。本来可以直接写function,然后打印,但是这样太麻烦。于是使用 # 可以很好的把期望的代码直接写的字符串,转移成C的字符串。
#define test_case_run(call) printf("%s\n", #call);
test_cases_run(void)
{
test_case_run(test_case_01_openfile);
test_case_run(test_case_02_writefile);
}
注意这里不需要实现 test_case_01_openfile 等函数就能够直接打印出来。并且,就算后期实现了 test_case_01_openfile 函数,该部分也不影响。
举例2
如果实现了该部分函数,还能通过修改宏定义,进而直接执行测试用例,具体如何修改?
void test_case_01_openfile(void)
{
printf("%s called\n", __func__);
}
#define test_case_run(call) do {\
printf("%s\n", #call);\
call(); \
} while(0)
test_cases_run(void)
{
test_case_run(test_case_01_openfile);
test_case_run(test_case_02_writefile);
}
这样实现函数之后还能调用.
# 的大概原理分析
# 在 C 语言编译预处理阶段,应该会自动遇到#的时候 会直接把后面的转义成为定义一个字符串,并且在全局静态段。从而在使用的时候直接可以调用这个字符串。注意#include不会
也就是 #abc就相当于 “abc"
但是必须是需要用宏定义里面用,做了一个实验,证明直接使用不会去创建一个全局的abc:
#include "stdio.h"
int main(void)
{
printf("%s\n", #abc);
return 0;
}
编译报错:
但是如果是在宏定义中:
#include "stdio.h"
int main(void)
{
#define tostr(m) #m
printf("%s\n", tostr(abc));
return 0;
}
运行后效果:
所以编译器原理不是直接根据# 来判断是否是字符串模式,而是进入#define的编译预处理阶段,在#define的语法树流程中进行字符串替换。所以# 可以说是 #define 的一个子功能
结论
C 语言中 使用 # 替换成字符串,必须在宏定义中才能使用。