C Primer Plus--C预处理器和C库(1)
编译程序之前,先由预处理器检查程序(因此称为预处理器)。根据程序中使用的预处理器指令,预处理用符号缩略语所代表的内容替换程序中的缩略语。
预处理器可以根据你的请求包含其他文件,还可以让编译器处理哪些代码。预处理器不能理解C,它一般是接受一些文本并将其转换成其他文本。-- (C Primer Plus中文第五版)
graph TD;
写好的C文件 --> 编译器翻译,为预处理做准备
编译器翻译,为预处理做准备 --> 预处理器寻找肯能存在的预处理指令,开始预处理
预处理符号
明显常量 #define
#define定义的作用域从定义出现的位置开始直到文件的结尾。
这里表明在函数里使用#define,凡是在它定义的后面的所有函数都是可以用的:
#include <stdio.h>
void pp();
int main() {
#define FF 10
pp();
}
void pp(){
printf("FF: %d ",FF);
}
输出:
FF: 10
要时刻记住:#define会把它之后的文件里的所有符合要求的文本替换掉。
每个#define行由三部分组成:
- 指令自身
#define - 缩略语,即宏(macro)
宏的名字中不允许有空格,必须遵循C变量名规则 - 替换列表或者主体(body)
预处理器在程序中发现了宏的实例后,总会用实体代替该宏。从宏变成最终的替换文本的过程称为宏展开(macro expansion)。
宏可以分两种:
- 类对象宏 object-like macro
宏用来代表值 - 类函数宏 function-like macro
外形和作用都与函数相似
#define TWO 2;
#define PX printf("X is: %d\n",x)
#define FMT "X+1 is: %d\n"
int main() {
int x =TWO;
PX;
printf(FMT,x+1);
}
输出:
X is: 2
X+1 is: 3
上面的程序其实被预处理器改成了:
int x = 2;
printf("X is: %d\n",x);
printf("X+1 is: %d\n",x+1);
//先变成int y = TWO * TWO;
int y = 2*2;
printf("y is: %d\n",y);
注意,宏展开过程中,是进行替换,并不进行计算。C编译器在编译时对所有常量表达式(只包含常量的表达式)求值,所以实际相乘过程发生在编译阶段,而不是预处理阶段。预处理器不进行计算,它只是按照指令进行文字替换操作。
宏展开过程中,会用宏的等价(即body)来替换文本,如果宏的body本身还含有宏的话,会继续展开这些宏。但是,双引号中的与宏缩略语一样的字符串无法被替换。
重定义常量
假设一个缩略语被定义后又在同文件中被定义,这样被称为重定义(redefinng a constant)。有的编译器会对这样提出警告,但允许重定义存在,有的则直接报错。
#define SIX 3 * 3
#define SIX 3 * 3
//上面这样的重定义会被编译器认为是重复定义,是相同的
#define SIX 3*3
//这样的重定义与上面两种是不同的
在#define中使用参数
类函数宏的定义中,用圆括号阔气一个或多个参数,随后这些参数出现在替换部分。
#define SQUARE(X) X*X
//一个参数X
//使用
int y = SQUARE(2);
宏调用和函数调用存在着区别:
程序运行时,函数调用把参数的值传递给函数,而编译前,宏调用把参数的语言符号传递给程序,仅仅是替换字符,而不计算。
#define SQUARE(X) X*X
int main() {
int x = 2;
int y = SQUARE(x);
printf("SQUARE(x) is: %d\n",y);
printf("SQUARE(x+2) is: %d\n",SQUARE(x+2));
}
输出:
SQUARE(x) is: 4
SQUARE(x+2) is: 8
按理说square(2+2)应该是16啊,怎么会是8呢?原来像刚才上面说的,预处理只是替换,因此SQUARE(x+2)中的X被x+2替换,最后成了x+2*x+2。*优先级高,,因此程序运行时先计算2*x,再加上x和2,也就成了8。要想实现平方的效果,需要重新定义:
#define SQUARE(X) ((X)*(X))
即使这样定义,还是无法避免自增、自减情况下的错误:
int x = 3;
int a = SQUARE(++x);
这里替换成++x字符后,进行了两次增量运算,最后结果肯定不是平方了。因此,在宏中不要使用增量或减量运算符。而且一定要充分的使用圆括号来保证正确的运算顺序。
在类函数宏中使用#运算符
上面说了在引号表示的字符串无法替换掉宏参数,,但是使用#预处理运算符,可以把传入的参量转化为文本替换到字符串里。
#define SQUARE(X) (X)*(X)
#define PF(X) printf("The square of " #X " is : %d\n",SQUARE(X))
int main() {
int x = 10;
PF(x);
PF(2+4);
}
输出:
The square of x is : 100
The square of 2+4 is : 36
在宏中使用##运算符
##运算符把两个语言符号组合成单个语言符号:
#define XVAR(X) x ## X
int XVAR(2) = 11;//声明了一个标识符为x2的变量
可变宏:...和__VA_ARGS__
宏定义中参数列表的最后一个参数为省略号,预定义宏__VA_ARGS_就可以被用在替换部分,以代表省略号省略了什么。
#define PF(X,...) printf("Result " #X " : " __VA_ARGS__)
int main() {
PF(1,"%d\n",10);
PF(2,"%d's power is %d\n",4,16);
}
输出:
Result 1 : 10
Result 2 : 4's power is 16
省略号只能代替最后的宏参数。
#define WRONG(X, ... ,Y) #X #__VA_ARGS__ #Y)//错误
这里有个有趣的现象:一般想要打印字符串,字符串都得用双引号括起来,这里不用:
#define STRING(... ) #__VA_ARGS__
int main() {
printf(STRING(abcdefg));
}
输出:
abcdefg
C Primer Plus--C预处理器和C库(1)的更多相关文章
- C Primer Plus之C预处理器和C库
编译程序前,先由预处理器检查程序(因此称为预处理器).根据程序中使用的预处理器指令,预处理器用符号缩略语所代表的内容替换程序中的缩略语. 预处理器不能理解C,它一般是接受一些文件并将其转换成其他文本. ...
- C Primer Plus--C预处理器和C库(2)
目录 #include指令 头文件 其他指令 #undef 条件编译 内联函数 #include指令 #include <头文件.h>//在标准系统目录中寻找头文件 #include &q ...
- C预处理器和C库
#define #include #undef #ifdef #else #endif #if #elif #else #endif 预处理宏: p463 _ _fun_ _是预定义标识符(函数作用域 ...
- 第 16 章 C 预处理器和 C 库(可变参数:stdarg.h)
/*------------------------------------------------- varargs.c -- use variable number of arguments -- ...
- 第 16 章 C 预处理器和 C 库(string.h 库中的 memcpy() 和 memmove())
/*----------------------------------------- mems.c -- 使用 memcpy() 和 memmove() ---------------------- ...
- 第 16 章 C 预处理器和 C 库(qsort() 函数)
/*---------------------------------------- qsorter.c -- 用 qsort() 排序一组数字 --------------------------- ...
- 第 16 章 C 预处理器和 C 库(直角坐标转换极坐标)
/*------------------------------------- rect_pol.c -- 把直角坐标转换为极坐标 ---------------------------------- ...
- 第 16 章 C 预处理器和 C 库(预定义宏)
/*------------------------------------- predef.c -- 预定义宏和预定义标识符 ------------------------------------ ...
- 第 16 章 C 预处理器和 C 库(条件编译)
/*-------------------------------------- names_st.h -- names_st 结构的头文件 ----------------------------- ...
随机推荐
- Java 之 Session 包含验证码登录案例
需求: 1. 访问带有验证码的登录页面login.jsp 2. 用户输入用户名,密码以及验证码. 如果用户名和密码输入有误,跳转登录页面,提示:用户名或密码错误 如果验证码输入有误,跳转登录页面, ...
- SpringBoot上传文件报错,临时路径不存在
异常信息 报错日志: The temporary upload location [/tmp/tomcat.7957874575370093230.8088/work/Tomcat/localhost ...
- RTSP协议介绍 (转)
1. 实 时流协议RTSP RTSP[3]协 议以客户服务器方式工作,它是一个多媒体播放控制协议,用来使用户在播放从因特网下载的实时数据时能够进行控制,如:暂停/继 续.后退.前进等.因此 RTSP ...
- github hooks 配置教程 钩子搭建(实测通过,手把手教程)
tips:如果本文对你有用,请爱心点个赞,提高排名,让这篇文章帮助更多的人.谢谢大家!❤ 本人hooks搭建成功,全程参考JellyBool老师的视频教程,有不懂的可以先去看下这个视频,跟着操作.本文 ...
- ETL讲解
ETL是将业务系统的数据经过抽取.清洗转换之后加载到数据仓库的过程,目的是将企业中的分散.零乱.标准不统一的数据整合到一起,为企业的决策提供分析依据. ETL是BI项目重要的一个环节. 通常情况下,在 ...
- node.js 学习一
Node.js 是单进程单线程应用程序,但是通过事件和回调支持并发,所以性能非常高. 与PHP 相似 都是单进程. Node.js 的每一个 API 都是异步的,并作为一个独立线程运行,使用异步函数调 ...
- CentOS环境安装python3,pip
安装python3(编译安装) 1. 下载python安装包到指定目录 cd /usr/local/src 这里的目录如果不存在可以自己建,当然也可以自己放在自己想放的位置,只要等会儿安装的时候能找到 ...
- Python面向对象三要素-多态
Python面向对象3要素-多态 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.多态概述 OCP原则:多用“继承”,少修改. 继承的用途:在子类上实现对基类的增强,实现多态. ...
- jquery选择器之全选择器
在CSS中,经常会在第一行写下这样一段样式 * {padding: 0; margin: 0;} 通配符*意味着给所有的元素设置默认的边距.jQuery中我们也可以通过传递*选择器来选中文档页面中的元 ...
- MySQL与安全
说到MySQL数据库的安全性,可能有大量的相关话题,下面将对几个关键问题进行概括性描述. (1)安全的一般性因素.包括使用强密码,禁止给用户分配不必要的权限,防止SQL注入攻击. (2)安装步骤的安全 ...