本篇笔记分享两个冷门却又不冷门的预处理指令。冷门是因为我们平时可能比较少用(或者说我们的编码能力还没到需要注重这些细节的水平),不冷门是因为这几个指令在一些优秀的C代码中用得很多。比如在Linux内核代码中:
#error指令
#error 指令让预处理器发出一条错误信息,并且会中断编译过程。下面我们从Linux代码中抽取出来一小段代码并做修改得到示例代码:
这段示例代码很简单,当RX_BUF_IDX宏的值不为0~3时,在预处理阶段就会通过#error 指令输出一条错误提示信息:
“Invalid configuration for 8139_RXBUF_IDX”
下面编译看一看结果:
不清楚C语言编译的几个阶段的朋友可以阅读往期笔记:【C语言笔记】C语言的编译过程
#pragma指令
#pragma指令应该是预处理指令中最复杂,其用法很多。下面简单看一下常见用法。
1、#pragma pack
我们可以利用#pragma pack来改变编译器的对齐方式:
#pragma pack(n) /* 指定按n字节对齐 */
#pragma pack() /* 取消自定义字节对齐 */
下面依旧以示例来说明。我们在之前的笔记:【C语言笔记】经典、易错的结构体内存对齐问题中也有提到结构体对齐的问题,那篇笔记我们没有自定字节对齐,按照编译器默认的对齐方式,代码为:
运行结果为:
至于结果为什么是16,可以去看一下上一篇笔记的分析。
下面我们使用#pragma pack指令来指定对齐的字节数。
(1)指定按1字节对齐
运行结果为:
(2)指定2字节对齐
运行结果为:
可见,指定的对齐的字节数不一样,得到的结果也不一样。指定对齐有什么用呢,大概就是可以避免了移植过程中编译器的差异带来的代码隐患吧。比如两个编译器的默认对齐方式不一样,那可能会带来一些bug。
2、#pragma message
该指令用于在预处理过程中输出一些有用的提示信息,如:
运行结果为:
如上,我们平时可以在一些条件编译块中加上类似信息,因为在一些宏选择较多的情况下,可能会导致代码理解起来会混乱。不过现在一些编译器、编辑器都会对这些情况进行一些很明显的区分了,比如哪块代码没有用到,那块代码的背景色就会是灰色的。
3、#pragma warning
该指令允许选择性地修改编译器警告信息。
例子:
#pragma warning( disable : 4507 34; once : 4385; error : 164 )
等价于:
#pragma warning(disable:4507 34) // 不显示4507和34号警告信息
#pragma warning(once:4385) // 4385号警告信息仅报告一次
#pragma warning(error:164) // 把164号警告信息作为一个错
这个指令暂且了解这么多,知道有这么一回事就可以。
关于#pragma指令还有很多用法,但比较冷门,这里暂且不列举,有兴趣的朋友可以自行学习。
以上就是本次的笔记分享,欢迎转发!