C语言边角料4:利用_Pragma来温柔的废弃API
扫描二维码
随时随地手机看文章
-
一、前言
-
二、操作过程
-
三、 _Prama 其他用法
一、前言
想象一下这个工作场景:你在为一个项目写一个功能库,别人调用库中提供的函数,后来你发现库里的函数A是多余的。
二、操作过程
1. 第一个版本的库
测试文件只有 3 个:api.h, api.c 和 main.c
api.h 文件内容:声明了 2 个函数。
- api.h 和 api.c: 库文件,编译得到 libapi.so;
- main.c:生成可执行程序,利用了上面生成的库 libapi.so;
main.c 文件内容:
以上代码的简单程度,等价于 helloworld 了。
2. 第二个版本的库
现在,你觉得 init 这个函数是多余的,想把它去掉,可以这么来修改。
既然 api.c 文件已经把这个函数删除了,但是 main.c 文件中又调用了这个函数,因此以宏定义的形式提供 init 这个符号。
(1) 是在宏替换时的表达式。因为这个函数可能被用在 if 条件判断中,因此需要返回一个值。在编译可执行文件时,编译器输出下面的这段话:
API_DEPRECATED 是另一个宏定义,扩展开来后就是让编译器在编译可执行程序时,打印出一段提示信息。
这样就达到了最初的目的!也就是提示使用者:这个函数已经被废弃了,最好别用它!
三 _Prama 其他用法
_Pragma 类似于 Microsoft 特定的 __pragma 关键字,只不过它是标准的一部分。它是在 C99 中为 C 引入的。对于 c ,它是在 c 11 中引入的。它允许将指令放入宏定义中。
1. 处理头文件重复包含
在头文件中,为了防止被重复包含,一般有 3 种处理方式:
// 头文件内容
#endif
(2) 第二种处理方式
// 头文件内容
以上这 2 种方式都可以防止同一个头文件被重复包含,但是还是有一些区别的。
这种方式与第二种方式的区别是:
#pragma :是一条预处理的指令,用来向编译器传达语言标准以外的一些信息,不能使用在宏中;#pragma 是编译器的扩展,也就是说它是由编译器来决定的,也许编译器A支持,但是编译器B就不一定支持了,虽然这种可能性比较小。
_Pragma :是一个操作符,属于语言的标准,因此可以嵌套在宏中,就像上面示例中那样;
2. 输出编译信息
上面两行的内容输出信息是一样的,需要注意的是嵌套的双引号需要用反斜线去转义。