C语言预处理详解

预处理是什么

在我们写完C语言程序的时候当我们开始运行程序时,程序会经过预处理,编译,汇编,链接这些过程之后才会生成可执行程序,这里我们讲的是预处理,预处理是编译的第一个阶段,在这个阶段,将对源文件进行文本操作,像删除注释,打开头文件插入头文件的内容,例如这些头文件

# include<stdio.h>
# include<string.h>

除此之外,还有替换#define定义的宏,#define命令可以允许把一个名称指定成任何的所需文本

# define N 5
# define M 3

这里程序运行时将会把代码中M和N出现的地方替换成5或者3 这个就是宏替换

define定义常量

下面我们来讲一下define定义常量的基本语法

# define number 5//把5替换成number
# define stu student//这样把命名简化
# define CSAE break;case//在写case时自动加上break
# define DO_FOREVER for(;;)//更加形象化
# define MALLOC(number,type)\(type*)malloc(number*sizeof(type))
//如果要先的定义代码过长可以用\来表示增加一行

有了这些定义可以帮助我们在写代码的时候能减少一些代码量,也可以起到代码有着更好的阅读性的作用。
还有在定义的时候要不要加;号?
我们来看看下面代码

# define number 5;
# include<stdio.h>
int main()
{
int a = number;//替换之后变成int a = 5;;
return 0;
}

如果在定义的时候加了分号,在下面使用的时候就会有两个分号,编译器就会报错,程序无法运行,所以根据我们平时的代码习惯最好不要加;号

define定义宏

#define 机制包括了⼀个规定,允许把参数替换到⽂本中,这种实现通常称为宏(macro)或定义宏(define macro)。下⾯是宏的申明⽅式:

#define name( parament-list ) stuff

其中的 parament-list 是⼀个由逗号隔开的符号表,它们可能出现在stuff中。
注意

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

这段话通俗来说就是 parament-list 就像我们平时写函数里面的参数,而stuff就是我们函数里面的内容,但实质上宏函数和我们平常写的函数还是有所不同的,这里方便理解可以理解成这样。
举例

# define ADD(x) (x+x)

这个宏接收一个参数x,在运行程序后会讲把源程序里面的ADD(x)部分替换成 x + x,x的值是多少这个宏就计算两个相同的数相加是多少,我们在看看下面的代码

# define fun(x+1) (x * x)

假设我们这里x传5想想这个宏的的值是多少?按照我们通常的思维来看这个算出来的值是36,但如果我们运行这个宏,就会发现这个值不是我们想的那样,实际上的值是11,这是为什么呢?我们不妨来看看他是怎么替换的

# define fun(x+1) (x+1 * x + 1)

这就是他替换的过程,是原封不动的把x 替换成x+1,按照优先级来计算的结果就是11,怎么才能解决这个问题呢,很简单,打上括号就行了.

# define fun(x+1) ((x) + (x))

这样就保证了运算的顺序,实现了我们想要的结果

所以说说在写宏函数的时候,我们应该细心一点,用括号把运算优先级给弄清楚,防止出现像这种情况,防止出现歧义或者一些不可预料的情况

同时我们也要注意像前置++和后置++在宏函数中的使用

y+1;//不带副作用y还是原来的值
y++;//带有副作用,y在使用之后自增1

这种是非常危险的,假如我们在比较x和y两个数谁大的时候如果在代码中使用了前置++和后置++就会产生歧义,像这样

#define MAX(a, b) ( (a) > (b) ? (a) : (b) )
x = 5;
y = 8;
z = MAX(x++, y++);
printf("x=%d y=%d z=%d\n", x, y, z);

这里通过预处理得到

z = ( (x++) > (y++) ? (x++) : (y++))

首先x是5,y是8,先是5和8比较大小,完了之后x变成6y变成9,9比6大返回y的值,然后y自增变成10,最后打印的结果是z = 9,x = 6,y = 10

宏替换的规则

宏替换的规则
在程序中扩展#define定义符号和宏时,需要涉及⼏个步骤。

  1. 在调⽤宏时,⾸先对参数进⾏检查,看看是否包含任何由#define定义的符号。如果是,它们⾸先
    被替换。
  2. 替换⽂本随后被插⼊到程序中原来⽂本的位置。对于宏,参数名被他们的值所替换。
  3. 最后,再次对结果⽂件进⾏扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上
    述处理过程。
    注意:
  4. 宏参数和#define 定义中可以出现其他#define定义的符号。但是对于宏,不能出现递归。
  5. 当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索。

宏函数和函数的对比

和函数相⽐宏的劣势:

  1. 每次使⽤宏的时候,⼀份宏定义的代码将插⼊到程序中。除⾮宏⽐较短,否则可能⼤幅度增加程序
    的⻓度。
  2. 宏是没法调试的。
  3. 宏由于类型⽆关,也就不够严谨。
  4. 宏可能会带来运算符优先级的问题,导致程容易出现错。
    宏有时候可以做函数做不到的事情。⽐如:宏的参数可以出现类型,但是函数做不到。
    和函数相⽐宏的优势:
    宏通常被应⽤于执⾏简单的运算。
    ⽐如在两个数中找出较⼤的⼀个时,写成下⾯的宏,更有优势⼀些。
#define MAX(a, b) ((a)>(b)?(a):(b))

那为什么不⽤函数来完成这个任务? 原因有⼆:

  1. ⽤于调⽤函数和从函数返回的代码可能⽐实际执⾏这个⼩型计算⼯作所需要的时间更多。所以宏⽐ 函数在程序的规模和速度⽅⾯更胜⼀筹。
  2. 更为重要的是函数的参数必须声明为特定的类型。所以函数只能在类型合适的表达式上使⽤。反之 这个宏怎可以适⽤于整形、⻓整型、浮点型等可以⽤于 > 来⽐较的类型。宏的参数是类型⽆关的。

我们看看下面代码,我们如果想要开辟一个float,或者int类型的动态内存,一般是这样写的

int*ptr = (int*)malloc(n*sizeof(int));
float*ptr = (float*)malloc(n*sizeof(float));

这样就比较麻烦,那有什么办法把他能不能写成一个函数呢直接传类型和想要的大小就可以了?
这里就可以用到宏函数了,他的参数就可以是类型,但函数不得行。

#include <stdio.h>
#include<stdlib.h>
# define MALLOC(number,type) (type*)malloc(number*(sizeof(type)))
int main()
{int* ptr = MALLOC(5, int);//替换之后变成int *ptr = (int*)malloc(5*sizeof(int));for (int i = 0; i < 5; i++){ptr[i] = i;}for (int i = 0; i < 5; i++){printf("%d ", ptr[i]);}return 0;
}

这样就会显得比较方便,也实现了我们想要的功能。

#和##

#运算符
#运算符将宏的⼀个参数转换为字符串字⾯量。它仅允许出现在带参数的宏的替换列表中。
#运算符所执⾏的操作可以理解为”字符串化“。
这段话有点抽象,我们可以写一个代码来感受一下

# define  PRINT(a) printf("the value of "#a " is %d", a);
int main()
{int a = 5;PRINT(a);return 0;
}

在这里插入图片描述
通俗来说就是就是把a这个字符原封不动的搬过去而不是a所代表的值
PRINT(a);//当我们把a替换到宏的体内时,就出现了#a,⽽#a就是转换为"a",时⼀个字符串
代码就会被预处理为:

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

##运算符

可以把位于它两边的符号合成⼀个符号,它允许宏定义从分离的⽂本⽚段创建标识符。 ## 被称为记号粘合这样的连接必须产⽣⼀个合法的标识符。否则其结果就是未定义的。

这⾥我们想想,写⼀个函数求2个数的较⼤值的时候,不同的数据类型就得写不同的函数。

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

这样就会比较复杂,我们可以写一个宏函数看看

#define COMPARE(type)    \type type##_max(type x, type y) \{   \return x>y?x:y;\}
//定义函数
COMPARE(int);//int_max
COMPARE(float);//float_max
int main()
{int r1 = int_max(3, 10);printf("%d\n", r1);float r2 = float_max(3.12f, 16.5f);printf("%.2f\n", r2);return 0;
}

#undef

这是一个移除宏定义的指令

# define MAX 5
int main()
{#undef MAX	int a = MAX;printf("%d", a);return 0;
}

条件编译

在编译⼀个程序的时候我们如果要将⼀条语句(⼀组语句)编译或者放弃是很⽅便的。因为我们有条
件编译指令。如果我们有些代码不要了,删了的话又不值得,这时候就可以用条件编译

# define DEBUG
int main()
{int a = 9;
#ifdef DEBUGprintf("%d", a);
#endifreturn 0;
}

这里我们就用到了条件编译,首先定义了一个debug 来判断a有没有成功打印 下面用了一个#ifdef DEBUG 这里表示如果定义了DEBUG就执行下面的语句**#endif和#ifdef连用**
常见的条件编译,用法有点类似于if else 语句

#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 OPTION1unix_version_option1();#endif#ifdef OPTION2unix_version_option2();#endif#elif defined(OS_MSDOS)#ifdef OPTION2msdos_version_option2();# endif# endif

头文件的包含

本地文件的包含

# include"test.h"

这种写法一般是我们自己写的头文件,在编译器中有一种查找策略
查找策略:先在源⽂件所在⽬录下查找,如果该头⽂件未找到,编译器就像查找库函数头⽂件⼀样在
标准位置查找头⽂件。
如果找不到就提⽰编译错误。

库文件的包含
这个就是C语言标准库里面的文件

查找头⽂件直接去标准路径下去查找,如果找不到就提⽰编译错误。

我们想一想如果标准库的头文件像我们自己写的头文件的方式去写可行吗?答案是可行的

# include"stdio,h"

但这种方式查找效率会底低下当然这样也不容易区分是库⽂件还是本地⽂件了。

嵌套文件包含

我们已经知道, #include 指令可以使另外⼀个⽂件被编译。就像它实际出现于 #include 指令的
地⽅⼀样。
这种替换的⽅式很简单:预处理器先删除这条指令,并⽤包含⽂件的内容替换。
⼀个头⽂件被包含10次,那就实际被编译10次,如果重复包含,对编译的压⼒就⽐较⼤。
那怎么防止这种情况出现,我们可以用这个命令

#pragma once

或者

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

这样就不会重复调用头文件了
完。
如果有什么错误欢迎指正!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

传参的指针,你的值到底变了没?!(有关函数形参修改的另类案例)

我们都知道&#xff0c;想要在函数中修改某个变量的值&#xff0c;传变量本身是没有用的。原因在于不同的函数在不同的空间上&#xff0c;函数的生命周期随着函数的调用而结束&#xff0c;因此在函数内部进行的值操作是不会对函数外的变量产生影响的。所以在函数里面想要修改变…

C语言使用STM32开发板手搓高端家居洗衣机

目录 概要 成品效果 背景概述 1.开发环境 2.主要传感器。 技术细节 1. 用户如何知道选择了何种功能 2.启动后如何进行洗衣 3.如何将洗衣机状态上传至服务器并通过APP查看 4.洗衣过程、可燃气检测、OLED屏显示、服务器通信如何并发进行 小结 概要 本文章主要是讲解如…

C语言-写一个宏,可以将一个整数的二进制位的奇数位和偶数位交换。

0xaaaaaaaa...等是什么&#xff1f;-CSDN博客https://blog.csdn.net/Jason_from_China/article/details/137179252 #define _CRT_SECURE_NO_WARNINGS 1 #include<stdio.h> #define SWAP(num) (((num & 0xAAAAAAAA) >> 1) | ((num & 0x55555555) << …

【C++】C到C++的入门知识

目录 1、C关键字 2、命名空间 2.1 命名空间的定义 2.2 命名空间的使用 2.2.1 加命名空间名称及作用域限定符 2.2.2 使用using将命名空间中某个成员引入 2.2.3 使用using namespace 命名空间名称引入 3、C输入&输出 4、缺省参数 4.1 缺省参数的概念 4.2 缺省参数的…

名字真的会影响我们的职业吗?

名字是我们身份的一部分&#xff0c;人们往往喜欢将一个人名字同一种具体的职业联系在一起&#xff0c;而如果这个人名字看上去更适合一项工作&#xff0c;那么他&#xff08;她&#xff09;就有更多的机会得到这份工作。因此&#xff0c;我们在给福主取名改名时也会与职业特点…

10.Python异常处理

为增强程序的健壮性&#xff0c;我们也需要考虑异常处理方面的内容。例如 &#xff0c;在读取文件时需要考虑文件不存在、文件格式不正确等异常情况。这 就是本章要介绍的异常处理。 1 第一个异常——除零异常 在数学中&#xff0c;任何整数都不能除以0&#xff0c;如果在计算…

PCL 计算点与圆的距离(3D)

文章目录 一、简介二、实现代码三、实现效果参考资料一、简介 3D中的圆可以有圆心、半径以及法线来进行表示,如下图所示: 这里我们假设: Δ = P − C \Delta=P-C Δ

Wasm初上手

总之也是为了扩宽技术面吧。。。我也不知道为什么就想试试了&#xff0c;就酱。 参考阅读&#xff1a;极客时间《WebAssembly入门课》 安装wasm的编译器Emscripten。Emscripten 是一个“源到源”语言编译器工具集&#xff0c;这个工具集可以将 C/C 代码编译成对应 JavaScript 代…

CSS使用clip-path实现元素动画

前言&#xff1a; 在日常开发当中&#xff0c;如果想要开发多边形&#xff0c;一般都需要多个盒子或者伪元素的帮助&#xff0c;有没有一直办法能只使用一个盒子实现呢&#xff1f; 有的&#xff1a;css裁剪 目录 前言&#xff1a; clip-path到底是什么&#xff1f; clip-pa…

【御控物联】 JavaScript JSON结构转换(4):对象To对象——规则属性重组

文章目录 一、JSON结构转换是什么&#xff1f;二、术语解释三、案例之《JSON对象 To JSON对象》四、代码实现五、在线转换工具六、技术资料 一、JSON结构转换是什么&#xff1f; JSON结构转换指的是将一个JSON对象或JSON数组按照一定规则进行重组、筛选、映射或转换&#xff0…

赋值语句还能当判断条件?涨芝士了!

赋值和条件看似是C语言中毫不相关的两个概念&#xff0c;虽然实际过程中我猜测不会有太多这种不太符合常理的情况出现&#xff0c;但是现在在学习的过程中&#xff0c;为了出题而出题总是会整出一些花活出来.....这很难不让人联想起高中时一些大佬为了彰显自己的数学天赋而自己…

树莓派串口读取陀螺仪ky9250(mpu9250)数据

9轴姿态角度传感器&#xff0c;其中ky9250陀螺仪由于自带卡尔曼动态滤波算法方便用户使用。ky9250陀螺仪基本可以在各个平台上进行数据的读取&#xff08;如stm32\arduino\C#\Matlab\树莓\Unity3d\python\ROS\英飞凌\Nvidia jetson linux 等&#xff09; 1、树莓派和ky9250的接…

AcWing刷题-区间合并

校门外的树 区间合并&#xff1a; from typing import List def merge(intervals: List[List[int]]) -> List[List[int]]:# 按照第一个元素从小到大进行排序intervals.sort(keylambda x: x[0])# 初始化一个新的数组new_list list()for i in intervals:# 把第一个数组元素添…

基于ssm旅游资源网站(java项目+文档+源码)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于ssm的旅游资源网站。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 旅游资源网站的主要使用者分为管理…

机器学习每周挑战——旅游景点数据分析

数据的截图&#xff0c;数据的说明&#xff1a; # 字段 数据类型 # 城市 string # 名称 string # 星级 string # 评分 float # 价格 float # 销量 int # 省/市/区 string # 坐标 string # 简介 string # 是否免费 bool # 具体地址 string拿到数据…

【42 可视化大屏 | 某瓣电影Top250数据分析可视化大屏】

文章目录 &#x1f3f3;️‍&#x1f308; 1 普版大屏&#x1f3f3;️‍&#x1f308;2 Flask版大屏&#x1f3f3;️‍&#x1f308;3 FlaskMysql版大屏&#x1f3f3;️‍&#x1f308; 4. 可视化项目源码数据 大家好&#xff0c;我是 &#x1f449;【Python当打之年(点击跳转)…

渐变色x轴换行柱状图

// 系统上云率const optionBar {title: {text: 系统上云率,left: left,textStyle: {color: "#fff",fontSize: 14,fontWeight: 650,align: "center",},},color: [#32C5FF, #00F766, #EECB5F],grid: {top: 40,bottom: 0,},legend: { // 控制图例组件show: …

数据结构·二叉树(2)

目录 1 堆的概念 2 堆的实现 2.1 堆的初始化和销毁 2.2 获取堆顶数据和堆的判空 2.3 堆的向上调整算法 2.4 堆的向下调整算法 2.4 堆的插入 2.5 删除堆顶数据 2.6 建堆 3 建堆的时间复杂度 3.1 向上建堆的时间复杂度 3.2向下建堆的时间复杂度 4 堆的排序 前言&…

【第十二届“泰迪杯”数据挖掘挑战赛】【2024泰迪杯】B题基于多模态特征融合的图像文本检索—解题全流程(论文更新)

【第十二届“泰迪杯”数据挖掘挑战赛】【2024泰迪杯】B题基于多模态特征融合的图像文本检索更新&#xff08;论文更新&#xff09; ​ 本节主要更新了论文、训练日志的log数据提取&#xff08;Loss、ACC、RK&#xff09;等数据可视化作图的代码 B题交流QQ群&#xff1a; 4583…

《乱弹篇(26)更好的自己》

俄乌、以巴、中东&#xff0c;烽火连天&#xff0c;持久酣战&#xff0c;搅得地球村住民不得安宁。虽说孰是孰非自有公论&#xff0c;但时评文难写也是评论界的普遍认知&#xff0c;所以今天笔者自觉地绕开时政话题&#xff0c;尽本“人民体验官”义务&#xff0c;推广人民日报…