【C语言进阶】预处理详解

引言

对预处理的相关知识进行详细的介绍

 158c3f50b199454985017a51dbef9841.png               ✨ 猪巴戒:个人主页✨

               所属专栏:《C语言进阶》

        🎈跟着猪巴戒,一起学习C语言🎈

目录

引言

预定义符号

#define定义常量

#define定义宏

带有副作用的宏参数

宏替换的规则

宏函数的对比

#和## 

#运算符

##运算符

命名约定

#undef

条件编译

头文件的包含

嵌套文件的包含


预处理又叫预编译,预处理是编译过程中的第一个步骤,主要是处理以#开头的预编译指令。

        

预定义符号

C语言设置了一些预定义符号,可以直接使用,预定义符号也是在预处理期间处理的。

__FILE__    //进行编译的源文件的文件名
__LINE__    //文件当前的行号
__DATE__    //文件被编译的日期
__TIME__    //文件被编译的时间
__STDC__    //如果编译器崔寻ANSI C,其值为1,否则就是未定义。
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
int main()
{
	printf("%s\n", __FILE__);
	printf("%d\n", __LINE__);
	printf("%s\n", __TIME__);
	printf("%s\n", __DATE__);

	return 0;
}

1c8f891dc5774057a6204866239de93a.png

         

#define定义常量

#define name stuff

在预编译的过程中,会用stuff来代替name。

比如下面的MAX就是要被替换的name,在预处理的阶段,MAX会被替换成100。

#define MAX 100
int main()
{
	printf("%d\n", MAX);
	return 0;
}

后面的stuff不仅可以是数字,也可以是一个长句。

#define PRINT printf("hello world!\n")
int main()
{
	PRINT;
	return 0;
}

65c157af11324fc2a81e2654eab31d37.png

        

#define定义宏

#define定义宏,不仅可以完成语句的替换,还可以将参数传进去。

#define name(parament_list) stuff

parament_list表示的就是参数,有了参数的加入,stuff的表达更加丰富。

注意:参数列表的左括号必须与name紧邻,如果两者之间有任何空白存在,参数列表就会被解释为stuff的一部分

比如说:

#define SQUARE(x) x+x
int main()
{
	int a = 10;
	printf("%d\n", SQUARE(a));
	return 0;
}

这里的SQUARE(x)中的 x 就是参数,将参数a代替x,然后带入后面的表达式。

这里的SQUARE(a),经过替换就是 a+a. 

043b70eb480d4d0b8234a1c670045981.png

但是这种定义宏的写法是存在瑕疵的,#define定义宏的书写一定要在各个参数上加上括号,在整体的表达式加上括号。

下面的书写方式才是正确的。

#define SQUARE(x) ((x)+(x))

接下来看两个错误的示范:

1.

#define SQUARE(x) x*x
int main()
{
	int a = 10;
	printf("%d\n", SQUARE(a+1));
	return 0;
}

这里的本来预想的结果是11*11=121,但是预处理并不是首先将参数进行计算,而是简单地将参数进行替换,SQUARE(a+1)会被替换成a+1*a+1(10+1*10+10),所以最后地结果是21。

解决办法就是

#define SQUARE(x) (x)*(x)

2.

#define SQUARE(x) x+x
int main()
{
	int a = 10;
	printf("%d\n", 10*SQUARE(a));
	return 0;
}

这里本来预想的结果是200,但是将参数进行替换,10*SQUARE(a)->10*a+a,结果就是110.

解决办法就是

#define SQUARE(x) (x+x)

总结:

为了避免宏定义中,参数中操作符或临近操作符之间不可预料的相互作用。我们应该毫不吝啬地在参数和参数表达式整体加上括号。

        

带有副作用的宏参数

刚刚讲了定义宏是什么,接下来是带有副作用的宏参数。如果参数带有副作用,那么你在使用这个宏的时候就可能出现危险,导致不可预测的后果。副作用就是表达式求值的时候出现的永久性效果。

x+1; //不带副作用
x++; //带有副作用

多次使用宏定义,那就会导致x的值发生变化,最终导致的结果就会大相径庭。

#define MAX(a,b) ((a)>(b)?(a):(b))
int main()
{
	int x = 4;
	int y = 5;
	int z = MAX(x++, y++);
	printf("%d %d %d\n", x, y, z);
	return 0;
}

fc84fbb0eb77484e91f728d35a671bd7.png

 这里x++,y++作为参数,MAX(x++,y++)会被替换成((x++)>(y++)?(x++):(y++)),首先x++和y++比较大小,x的值会变成5,y的值会变成6,那么后面的y++表达式就会被执行,因为是前置++,先执行后++,所以结果是6,但是y的值变成了7。

        

宏替换的规则

在程序中扩展#define定义符号和宏时,需要涉及几个步骤。

  1. 在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果时,它们首先被替换。
  2. 替换文本随后被插入到程序中原来文本的位置。对于宏,参数名被它们的值所替换。
  3. 最后,再次对结果文件进行扫描,看看它是否包含任何由#define定义的符号。如果时,就重复上述处理过程。

注意:

宏参数和#define定义中可以出现其他#define定义的符号。但是对于宏,不能出现递归。

当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索。

        

宏函数的对比

属性#define定义宏函数
代码长度每次使用时,宏代码都会被插入到程序中。除了非常小的宏之外,程序的长度会大幅度增长函数代码只出现与一个地方,每次执行函数调用的都是同一个地方
执行速度更快存在函数的调用和返回函数栈帧的时间,速度会慢一些
操作符优先级宏参数的求值是在所有周围表达式的上下文环境里,除非加上括号,否则邻近操作符的优先级可能会产生不可预料的后果,所以建议宏在书写的时候多些括号。函数参数只在函数调用的时候直接将参数的值传递给函数。
带有副作用的参数参数可能被代替到宏体中的多个位置,如果宏的参数被多次计算,带有副作用的参数求值可能会产生不可预料的结果函数参数只在传参的时候对参数进行求值
参数类型宏的参数与类型无关,只要对参数的操作是合法的,它就可以使用与任何参数类型函数的参数是与类型有关的,不同的参数类型就要不同的函数,所以比较严格
调试宏是不方便调试的函数是可以逐语句调试
递归宏是不能递归的 函数是可以递归的

         

#和## 

#运算符

#运算符将宏的一个参数转换为字符串字面量,它仅允许出现在带参数的宏的替换列表中。

#运算符所所执行的操作可以理解为“字符串化”。

#define PRINT(n) printf("the value of "#n"is %d",n)

之所以通过#运算符进行字符串化,是因为在字符串中,#define是不会进行替换的。

printf("the value of ""a""is %d",a);

##运算符

##运算符可以把位于它两边的符号合成一个符号,它允许宏定义从分离的文本片段创立标识符。##被称为记号粘连。

int int_max(int x, int y)
{
	return x > y ? x : y;
}

float float_max(float x, float y)
{
	return x > y ? x : y;
}

由于类型的不同,这个比较大小的功能必须分为两个函数进行实现。

下面是一个定义宏,通过下面的定义宏将类型作为参数,替换函数中的类型。

其中\是换行符,type_max如果直接书写就是函数名称了,分开书写,然后用##来粘合,就可以实现替换函数名。

#define GENERIC_MAX(type)\
type type##_max(type x,type y)\
{\
    return (x>y?x:y);\
}

GENERIC_MAX(int)
GENERIC_MAX(float)//等同于上面两个函数的代码

         

命名约定

一般来说函数和宏语法很相似,所以语言本身没法帮助我们区分二者。

那我们平时的一个习惯是

  • 把宏名全部大写
  • 函数名不全部大写

        

#undef

我们直到#define name就是定义一个常量,#undef就是取消宏定义。

#define MAX 100
#undef MAX     //这里往后MAX就取消宏定义,不能直接使用MAX了

        

条件编译

条件编译指令有

#ifdef #if #endif #elif #else #ifndef

#if 表达式 语句: 后面加表达式,和if的意思一样,但是这个步骤是在预处理的时候完成的,如果表达式为假,就不会出现在后面的文件中。

#elif 表达式 语句:和else if的意思一样,和#if搭配使用。

#else 语句:和else的意思一样。

#endif :不同的是我们在使用完条件编译指令后,要加上#endif,表示预处理完成。

#ifdef  :  意思:如果后面被定义,就执行后面的语句。

#ifndef : 意思:如果后面没被定义,就执行后面的语句。

int main()
{
    #ifdef MAX
        printf("haha\n");
    #endif
    return 0;
}
#if 常量表达式
        //...
#endif
//常量表达式由预处理器求值。
如:

#define __DEBUG__ 1
#if __DEBUG__
            //..
#endif

2.多个分⽀的条件编译
#if 常量表达式
            //...
#elif 常量表达式
            //...
#else
            //...
#endif

3.判断是否被定义
#if defined(symbol)
#ifdef symbol

#if !defined(symbol)
#ifndef symbol

4.嵌套指令
#if defined(OS_UNIX)
        #ifdef OPTION1
            unix_version_option1();
        #endif
        #ifdef OPTION2
            unix_version_option2();
        #endif
#elif defined(OS_MSDOS)
        #ifdef OPTION2
            msdos_version_option2();
        #endif
#endif

         

头文件的包含

#incldue <filename>

库函数通过<>包含,会在指定的标准头文件位置查找头文件。

#include "filename"

通过”“包含,会先在源文件所在目录下查找,如果头文件未找到,编译器就会像查找库函数文件的方式一样去标准的头文件位置查找。

所以库函数也可以用”“来包含,但是这样查找的效率就慢,也不容易区分库文件和本地头文件。

        

嵌套文件的包含

#include指令会将头文件进行包含,在预处理阶段实现,但是如果是多个源文件的交互合作,那么程序会对同一个头文件多次包含,那么就会在大大增加程序包含的代码量。

所以我们通过条件编译解决头文件多次引用问题:

ifndef __TEST_H__
#define __TEST_H__
//头⽂件的内容
#endif 

//或者
#pragma once

当头文件被第一次引用时,就会定义__TEST_H__,那么在第二次打开的时候,__TEST_H__就已经被定义,因为#ifndef,头文件就不会再次包含。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/342021.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

空气净化器or宠物空气净化器?五款猫用空气净化器优质推荐!

作为一个养猫家庭的主人&#xff0c;每天都要面对清理猫砂盘的挑战&#xff0c;这种令人难以形容的气味实在让人难以忍受。尤其是家里有小孩和老人&#xff0c;他们可能会出现过敏性鼻炎等问题&#xff0c;而抵抗力较差的人更容易受到影响。此外&#xff0c;换毛季节到来时&…

Qt 拖拽事件示例

一、引子 拖拽这个动作,在桌面应用程序中是非常实用和具有很友好的交互体验的。我们常见的譬如有,将文件拖拽到某个窗口打开,或者拖拽文件到指定位置上传;在绘图软件中,选中某个模板、并拖拽到画布上,画布上变回绘制该模板的图像… 诸如此类,数不胜数。 那么,在Qt中我…

【MySQL】最左匹配原则

最左匹配原则 0x1 简单说下什么是最左匹配原则 顾名思义&#xff1a;最左优先&#xff0c;以最左边的为起点任何连续的索引都能匹配上。同时遇到范围查询(>、<、between、like&#xff09;就会停止匹配。 例如&#xff1a;b 2 如果建立(a&#xff0c;b&#xff09;顺序…

远程桌面--虚拟机与主机的文件传输

注意&#xff1a; 确保VMware开头的服务全部在运行进入虚拟机打开文件管理器点击计算机右键选择属性在选择远程管理选择允许 1.winR 输入mstsc 2.输入虚拟机的ip地址 2.输入虚拟机的密码 上面的Administrator是虚拟机的用户名&#xff0c;有时会需要我们手动输入 3.验证…

MySQL连表操作之一对多

MySQL连表操作之一对多 目录 引入外键 Navicat创建外键使用外键SQL命令创建外键代码删除外键代码增加外键通过外键进行数据操作 正文 回到顶部 引入 当我们在数据库中创建表的时候&#xff0c;有可能某些列中值内容量很大&#xff0c;而且重复。 例子&#xff1a;创建一个…

基于SpringBoot Vue二手闲置物品交易系统

大家好✌&#xff01;我是Dwzun。很高兴你能来阅读我&#xff0c;我会陆续更新Java后端、前端、数据库、项目案例等相关知识点总结&#xff0c;还为大家分享优质的实战项目&#xff0c;本人在Java项目开发领域有多年的经验&#xff0c;陆续会更新更多优质的Java实战项目&#x…

C语言之反汇编查看函数栈帧的创建与销毁

文章目录 一、 什么是函数栈帧&#xff1f;二、 理解函数栈帧能解决什么问题呢&#xff1f;三、 函数栈帧的创建和销毁解析3.1、什么是栈&#xff1f;3.2、认识相关寄存器和汇编指令3.2.1 相关寄存器3.2.2 相关汇编命令 3.3、 解析函数栈帧的创建和销毁3.3.1 预备知识3.3.2 代码…

Python 猎户星空Orion-14B,截止到目前为止,各评测指标均名列前茅,综合指标最强;Orion-14B表现强大,LLMs大模型

1.简介 Orion-14B-Base是一个具有140亿参数的多语种大模型&#xff0c;该模型在一个包含2.5万亿token的多样化数据集上进行了训练&#xff0c;涵盖了中文、英语、日语、韩语等多种语言。在多语言环境下的一系列任务中展现出卓越的性能。在主流的公开基准评测中&#xff0c;Orio…

Qt —— 自定义飞机仪表控件(附源码)

示例效果 部署环境 本人亲测版本Vs2017+Qt5.12.4,其他版本应该也可使用。 源码1 qfi_ADI::qfi_ADI( QWidget *parent ) :QGraphicsView ( parent ),m_scene ( nullptr )

英语连读技巧12

1. this thing – 这件事 连读听起来就像是&#xff1a;【the sing】 连读的音标为&#xff1a; 例句&#xff1a;i cannot get this thing off my mind. 发音指导&#xff1a;注意 “this” 和 “thing” 两个词在连读时&#xff0c;“s” 和 “th” 的音接近融合&#xf…

浏览器提示“此网站的安全证书有问题”

有时候在浏览一些网站的时候&#xff0c;点进去的时候不是直接显示内容&#xff0c;而是弹出来一个警告的界面&#xff0c;告诉你此网站的安全证书有问题&#xff0c;浏览器是建议你不要再继续访问了&#xff0c;当然你也可以选择继续访问。那么&#xff0c;你还敢继续访问一个…

2.1.2 关系模式

1. 什么是关系模式 2. 定义关系模式 3. 关系模式与关系 1&#xff0e;什么是关系模式 关系模式是型&#xff0c;关系是值 。 关系模式是对关系的描述&#xff1a; 描述关系元组集合的结构 &#xff0c; 属性构成 l&#xff0c;属性来自的域 l&#…

品牌营销:长期价值与潜在利益的共赢之路

在当今品牌营销面临着前所未有的挑战。品牌信息的传递不仅需要迅速&#xff0c;更要持久。而在这个过程中&#xff0c;产品的长期价值和潜在利益成为品牌营销的核心。本文迅腾文化将通过奥卡姆剃刀定律和霍夫曼编码的视角&#xff0c;探讨品牌营销如何实现长期价值和潜在利益的…

gitlab.rb主要配置

根据是否docker安装,进入挂载目录或安装目录 修改此文件,我一般是在可视化窗口中修改,有时候也在命令行手敲 将下面的配置复制到该文件中 external_url http://192.168.100.50 # nginx[listen_port] = 8000 (docker安装的这一行不需要,因为端口映射导致此处修改会导致访问…

助力公益事业,吉林长春市第二社会福利院与清雷科技达成合作

“通过部署清雷科技智慧康养大屏和毫米波雷达监测设备&#xff0c;可以实时查看全院入住人员的生命体征情况&#xff0c;包括呼吸、心率、在离床状态、睡眠报告等&#xff0c;对呼吸异常、跌倒风险异常的人员还会特别标注提醒&#xff0c;提高护理员工作效率的同时&#xff0c;…

SCT9430TVBR:3.8V-36V输入,3A同步降压DCDC转换器

• 通过开关节点无振铃降低 EMI • 400KHz固定开关频率&#xff0c;6% 抖频拓展频谱 • 轻载条件下的脉冲跳跃模式 PSM • 3.8V-36V 宽输入电压范围 • 最大连续3A输出负载 • 0.8V 1% 反馈参考电压 • 集成80mΩ (Rdson) 上端MOSFET 和 42mΩ (Rdson) 下端MOSFET • 1uA 关断电…

【机组】指令控制模块实验的解密与实战

​&#x1f308;个人主页&#xff1a;Sarapines Programmer&#x1f525; 系列专栏&#xff1a;《机组 | 模块单元实验》⏰诗赋清音&#xff1a;云生高巅梦远游&#xff0c; 星光点缀碧海愁。 山川深邃情难晤&#xff0c; 剑气凌云志自修。 ​ 目录 &#x1f33a;一、 实验目…

Redis——RDB持久化

前言 Redis是一个键值对数据库服务器&#xff0c;服务器中通常包含任意个非空数据库&#xff0c;而每个非空数据库中又可以包含任意个键值对&#xff0c;为了方便起见&#xff0c;我们将服务器中的非空数据库以及它们的键值对统称为数据库状态。 因为Redis数据库是内存数据库&a…

如何在Docker上运行Redis

环境: 1.windows系统下的Docker deckstop 1.Pull Redis镜像 2.运行Redis镜像 此时,Redis已经启动&#xff0c;我们登录IDEA查看下是否连接上了 显示连接成功&#xff0c;证明已经连接上Docker上的Redis了

INTEWORK—PET 汽车软件持续集成平台

产品概述 INTEWORK-PET-CI是经纬恒润自主研发的汽车软件持续集成&持续交付平台&#xff0c;在传统的持续集成基础上深化了研运一体化&#xff08;DevOps&#xff09;的概念&#xff0c;将嵌入式软件中的拉取代码、检查、构建、测试、版本管理以及发布交付等环节串联起来&am…