关于我
欢迎来到我的博客!这里汇集了我对编程和技术的洞见和总结。本站内容分为几个主要类别,涵盖从具体技术实现到编程理念的广泛话题。
主要内容分类
- 项目工程:深入探讨技术的实现细节和理解。
- C/C++:围绕C/C++语言的技术点和编程技巧进行详细总结。
- 程序员哲学:分享程序员在职业生涯中应该具备的哲学理念和思考方式。
想要了解更多具体内容,您可以访问文章分类页面。
联系我
如果您有任何问题或想要交流,欢迎通过关于页面与我联系。
感谢您的阅读和支持,希望我的博客能为您的技术旅程带来帮助!
定义一个宏,来检测另一个宏是否被定义的优化代码
该问题的原提出者见stackoverflow:
https://stackoverflow.com/questions/26099745/test-if-preprocessor-symbol-is-defined-inside-macro
关于这个宏定义:
1 | #define TRACE(x) "" #x |
#x是c预处理器的一个操作符,将操作数转换位一个字符串字面量.
即有如此代码
1 | printf(TRACE(this_macro)); |
便会输出this_macro
这里的问题就是为何在#x前面加一个空字符串""
对于
1 | #define TRACE2(x) #x |
这两种不同的宏定义,具体表示如何,甩给AI试试.
AI结论:在大多数的时候,这两个都是相同的效果.只有在极少数情况下会出现差别.
claude表示:TRACE3(x)会更安全,因为它会拼接多个宏参数于一个字符串.而TRACE2(x)不会拼接字符串.
google Gemini 1.5 Pro表示:
1 | printf(TRACE3(hello) TRACE3(world)); |
TRACE3会连接这两个字符串,而TRACE2不行.
chatgpt4 表示TRACE3会在非法情况下仍然生成空字符串,更加安全.
理论上,我相信了AI给出的解答,但是我仍然编写测试集macro1.c,用于测试所有的情况.
macro1.c代码:
1 |
|
其他的情况,这两者并没有表示出不一样.于是我尝试了一种非法情况:
即使在对于TRACE2和TRACE3传入非法的参数x(x表示没有传入任何东西),理论上:TRACE2显然是没有任何字符串生成,这会导致ERROR,TRACE3会生成一个空字符串,显然TRACE3更安全.
但是在编译的时候,这两者都出现了一样的WARING:
1 | macro1.c: In function ‘main’: |
可能是编译器的优化导致的,于是我传入CFLAGS=-O0取消优化
1 | gcc macro1.c CFLAGS=-O0 |
结果仍然相同.
目前没有好的解释,不过问题不大,好,这个问题过.
但是显然,这个宏的功能很强大.
对于定义一个检测宏是否被定义的宏
Stackoverflow中(上述提到),第一个回答中定义了:
1 |
|
对上述代码描述:
如果一个宏被定义了,那么该宏会在TRACE_STRINGIFY宏中被展开,在"" #macro中不会被展开,保持原有的字符信息.
此时strcmp函数返回非0值,执行打印函数,表示该宏被定义.
反之,该宏不会在TRACE_STRINGIFY宏中被展开,而是被转换为原字符串字面值,这与"" #macro的作用一致,此时strcmp函数返回0,则if语句不会执行,打印函数不会被执行,没有任何的输出.
通过描述信息可以看出来,这个宏定义对于这种宏是无效的:
1 |
上述是c++版本,我在macro2.c程序中给出c版本
并且给出了另一种优化
macro2.c代码:
1 |
|
在macro2.c代码中,我将TRACE宏定义为:检查某一个宏,如果该宏没有被定义,则打印出错误信息,唯一的逻辑改动就是在strcmp函数前加非
根据上述原理,则可以实现定义一个宏,来检测另一个宏是否被定义,如果没有被定义,则输出错误信息.
打印错误信息的实现,是为TRACE宏添加c语言的变参系统,调用fprintf函数,将错误信息传入stderr,以及使用预定义宏__FILE__,__LINE__来标记错误位置.
添加错误信息的TRACE宏,适用于一些非配置选项类项目构建,用于检测某些需要的宏,如果没有该宏则输出错误信息.而没有添加错误信息的TRACE宏,适用于需要kconfig工具来配合进行需要配置选项类项目的构建,用于检测用户的配置选项,来有选择地编译某些文件或者不编译某些文件.
If you like this blog or find it useful for you, you are welcome to comment on it. You are also welcome to share this blog, so that more people can participate in it. All the images used in the blog are my original works or AI works, if you want to take it,don't hesitate. Thank you !