建设银行网站上不去,网站建设耂首先金手指,互动营销是什么,市场调研报告内容【C语言终章】预处理详解#xff08;上#xff09; 当你看到了这里时#xff0c;首先要恭喜你#xff01;因为这里就是C语言的最后一站了#xff0c;你的编程大能旅途也将从此站开始#xff0c;为坚持不懈的你鼓个掌吧#xff01; #x1f955;个人主页#xff1a;开敲…【C语言终章】预处理详解上 当你看到了这里时首先要恭喜你因为这里就是C语言的最后一站了你的编程大能旅途也将从此站开始为坚持不懈的你鼓个掌吧 个人主页开敲 所属专栏C语言 文章目录
1. 预定义符号
2. #define定义常量
3. #denfine定义宏
4. 带用副作用的宏参数
5. 宏替换的规则
6. 宏和函数的对比
7. #和## 7.1 #运算符 7.2 ## 运算符 1. 预定义符号 在C语言中的预处理阶段会处理预定义符号而C语言中设置了一些预定义符号我们可以直接使用 1 __FILE__ //当前文件的路径 2 __LINE__ //当前的行号 3 __DATE__ //当前程序编译瞬间的日期 4 __TIME__ //当前程序编译瞬间的时间 5 __STDC__ //如果编译器遵循ANSI C其值为1否则未定义 举个例子 在VS编译器中就不遵循ANSI C因此报错。
2. #define定义常量 #define name //名称 stuff //定义后的值 例如 1 #define MAX 100 //定义MAX为100 2 #define reg register //为 register 这个关键字创建一个简短的名字 3 #define do_forever for(;;) //为死循环创建一个名字 4 #define CASE break;case //在写case语句时自动把 break 加上 //如果define定义的语句过长可以用续行符分成几行写 5 #define DEBUG_PRINTF printf(file :%s\tline:%d\t\ date:%s\ttime::%s\n,\ __FILE__,__LINE__, \ //续行符 __DATE__,__TIME__) 问在#define定义标识符时后续需不需要加上;?
答加上纯属多余有时还会带来不好的后果 例如 这里我们在 100 加上了 ; 在打印时直接报错了这是为什么 因为MAX 是100因此在printf中它是这样的 1 printf(%d\n,100;); 在里面多了一个这显然是错误的语法。
3. #denfine定义宏 #define机制包括了一个规定允许把参数直接原样替换到文本中这种实现通常就称之为宏macro或者定义宏define macro。 下面是宏的声明方式 1 #define name(parament-list) stuff //parament-list 参数列表 注括号必须和name紧挨着如果中间有空格参数列表就会被视为stuff的一部分
举个例子 1 #define SQUARE(x) x*x 这个宏接收一个参数x如果你在程序中写下 SQUARE(5)那么这个宏最终就会被预处理器给替换为 5*5。 注这个宏存在一个问题下面来看一段代码 根据我们以往的知识这段代码最后输出的应当是 36因为在函数中这里传过去的值应当是6然后计算的就是6*6。下面来看输出结果 可以看到这里输出的居然是 11 这是为什么呢原因就是上面说的原样替换 像这样原样替换进去后表达式就变成了 51*51因此计算的结果就是11那如何解决这个问题呢很简单只需要在定义时加上括号即可 1 #define SQUARE(x) (x)*(x) 这里还有一个宏定义 1 #define ADD(x) (x)(x) 这个宏定义也是有着和上面一样的问题 这里我们本意是想输出120但这里却输出了66还是原样替换的原因 4. 带用副作用的宏参数 当宏参数在宏的定义中出现超过一次的时候如果参数带有副作用那么你在使用这个宏的时候就可能出现危险导致不可预测的后果。 1 x1; //并没有改变x本身的值不带有副作用 2 x; //改变了x本身的值带有副作用 下面来看一段代码 思考一下上面的代码输出的是什么。 可以看到最后输出时a和b的值都被改变了这也是上面所说的原样替换 5. 宏替换的规则 在程序中扩展#define定义符号和宏时需要涉及几个步骤
① 在调用宏时首先对参数进行检查看看是否包含任何由#define定义的符号。如果是他们首先被替换例如
② 替换文本随后被插入到程序中原来文本的位置。对于宏参数名被他们的值所替换。
③ 再次对结果文件进行扫描看看它是否包含任何由#define定义的符号。如果是那么就重复上述过程 例如 注意
① 宏参数和#define定义中可以出现其他#define定义的符号。但是对于宏不能因为宏无法递归。
② 当预处理器搜索#define定义的符号的时候字符串常量的内容并不被搜索例如 6. 宏和函数的对比 宏通常被用于执行简单的运算比如在两个数中找出较大的数就可以用下面的宏 1 #define MAX(a,b) (a)(b)?(a):(b) 那为什么不用函数来完成这个任务呢
有两点原因
① 用于调用函数和从函数返回的代码可能比实际执行这个小型计算机工作所需要的时间更多。所以宏在执行简单运算时在规模和速度方面都要优于函数。 ② 更为重要的是函数的参数不能以类型的形式传递所以函数只适合在类型合适的表达式上用。但是宏可以因为宏所做到的是原样替换因此在传参是宏并不会在乎你传的是什么它会直接把你传的参数给替换进宏中例如 前面说了和函数相比宏的优势接下来讲讲和函数相比宏的劣势
① 每次使用宏的时候一份宏定义的代码将插入到程序中。除非定义的宏比较短否则可能大幅增加程序的长度。
② 由于宏只是一条代码指令一次性就完成了好几条甚至更多的代码指令因此宏是没法进行调试的。
③ 宏由于类型无关因此也就不够严谨
④ 宏可能会带来运算符优先级的问题比如 5.宏替换规则 中的导致程序的可能会出错。
7. #和## 7.1 #运算符 #运算符将宏的一个参数转换为字符串字面量。注它仅允许出现在带参数的宏的替换列表中。 #运算符所执行的操作可以理解为 字符串化。 下面来看一段代码 当我们想用宏来实现这一段代码时应该如何做到呢
我们可能会想到这种方法 1 #define PRINT(a) printf(the value of a is %d\n,a) 显然上面这种方法是行不通的因为预处理器不会扫描字符串常量中的内容因此printf中的a并不会被替换那么这时候就需要我们用到#运算符了 这里使用#运算符将a字符串化变为字符串字面量这也预处理器就可以扫描a并将其替换。 7.2 ## 运算符 ##运算符可以把位于它两边的符号合成一个符号它允许宏定义从分离的文本片段创建标识符。##被称为记号粘合。 注这样的连接必须产生一个合法的标识符。否则其结果是未定义的。 这里我们想想在我们想要求两个数中的较大值时使用函数的话比较不同类型的数就得实现不同的函数比如 显然这样写起来太麻烦了现在我们学习了宏定义之后可以试着写一下宏 在实际的开发过程中##使用的很少因此很难举例出形象贴切的例子。 创作不易点个赞呗蟹蟹啦~