C语言之预处理详情

news2025/1/22 14:42:58

目录

  • 前言
  • 1.预定义符号
  • 2.#define定义常量
  • 3.#define定义宏
  • 4.带有副作用的宏参数
  • 5.宏替换的规则
  • 6.宏和函数的对比
  • 7.#和##运算符
    • 7.1 #运算符
    • 7.2 ##运算符
  • 8.命名约定
  • 9.undef
  • 10.命令行指令
  • 11.条件编译
  • 12.头文件的包含
    • 12.1 头文件包含方式
      • 12.1.1 本地头文件包含
      • 12.1.2 库文件包含
    • 12.2 嵌套文件包含
  • 13.其他预处理指令
  • 总结

前言

我们了解预处理有利于理解程序运行,自行查找问题以及代码修复。

1.预定义符号

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

__FILE__  //进行编译的源文件
__LINE__  //文件当前的行号
__DATE__  //文件被编译的日期
__TIME__  //文件被编译的时间
__STDC__  //如果编译器遵循ANSI C,其值为1,否则未定义

测试代码如下:

#include <stdio.h>
int main()
{
	printf(" __FILE__: %s\n", __FILE__);
	printf(" __LINE__: %d\n", __LINE__);
	printf(" __DATE__: %s\n", __DATE__);
	printf(" __TIME__: %s\n", __TIME__);
	//printf(" __STDC__: %D\n", __STDC__);
	return 0;
}

VS2022 X64输出结果如下
在这里插入图片描述
VS未定义__STDC__ .
在这里插入图片描述


2.#define定义常量

基本语法:

#define name stuff

测试代码如下

#include <stdio.h>
#define MAX 100//定义常量
#define reg register//为关键字register创建一个简短的名字
#define do_forever for(;;)//死循环,用更形象的符号来替换一种实现
#define CASE break;case//在写case语句的时候自动加上break

//定义的stuff过长,可以分成几行写,除最后一行,每行后面加续行符'\'
#define DEBUG_PRINT printf("file:%s\nline:%d\n\
							date:%s\ntime:%s\n",\
							__FILE__,__LINE__,\
							__DATE__,__TIME__)
int main()
{
	//CASE测试
	int input = 1;
	switch(input)
	{
	case 0:
		//
	CASE 1:
		//
		break;
}
	//测试
	DEBUG_PRINT;
	return 0;
}

VS2022 X64输出结果如下
在这里插入图片描述
注:#define定义标识符最好不要在末尾加";",因为#define name stuff就是将name换成stuff,如果是stuff;,原本的name就会被替换成stuff;

举个例子

#include <stdio.h>
#define M 100;
int main()
{
	int a = M;
	printf("%d\n", M);
	return 0;
}

上面的代码等价于下面的代码

#include <stdio.h>
int main()
{
	int a = 100;;//这个地方没问题,因为;等价于空语句
	printf("%d\n", 100;);//但是这个地方影响打印,会报错
	if(1)
		a = M;//这个地方也有问题,因为if后没有{},只能跟一条语句,现在是两条语句
	else
		a = 0;
	return 0;
}

VS2022 X64系统如下
在这里插入图片描述


3.#define定义宏

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

#define name(parament-list) stuff

其中的parament-list是一个由逗号隔开的符号表,它们可能出现在stuff中。
注意:
参数列表的左括号必须与name相邻,如果两者之间有任何空白存在,参数列表就会被解释为stuff的一部分。

举个例子:

#define SQUARE(x) x*x

该宏接收参数x,用x*x的值代替SQUARE(x).
测试代码如下:

#include <stdio.h>
#define SQUARE(x) x*x
int main()
{
	printf("%d\n", SQUARE(5));
	printf("%d\n", SQUARE(5+1));
	return 0;
}

VS2022 X64输出结果如下

25
11

其实宏替换的时候是直接替换,不会对传入的表达式进行计算,所以SQUARE(5+1)等价于5+1*5+1,因此第二行的结果是11.

所以我们要考虑在参数的外面带上括号,防止由于相邻操作符优先级的关系导致结果有误差。

对宏定义修改后,下面代码的输出结果是36.

#include <stdio.h>
#define SQUARE(x) (x)*(x)
int main()
{
	printf("%d\n", SQUARE(5+1));
	return 0;
}

最安全的方式其实是在最后结果也加上括号,如下代码所示。

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

比如下面这段代码输出结果是66,是10*(5+1)+(5+1),但我们想要的是10ADD(6),也就是1012=120.

#include <stdio.h>
#define ADD(x) (x)+(x)
int main()
{
	printf("%d\n", 10*ADD(5 + 1));
	return 0;
}

修改后的代码如下所示,输出结果是120.

#include <stdio.h>
#define ADD(x) ((x)+(x))
int main()
{
	printf("%d\n", 10*ADD(5 + 1));
	return 0;
}

综上所述,宏定义要注意在各个参数以及最终结果加上括号。


4.带有副作用的宏参数

比如前置、后置++(–)会对参数本身进行修改,所以我们在定义宏的时候,如果传入的参数是a++等内容,特别是多次替换的情况下,由于参数本身被修改,结果将会是不可预测的。

举个例子:

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

代码int z = MAX(x++, y++);等价于int z = ((x++)>(y++)?(x++):(y++));后置++是先使用后++,首先进行x与y的比较,x<y,且此时x变成6,y变成9,执行z = y++,z变成9,y+1变成10.

VS2022 X64输出结果如下

6 10 9

但是我们想要的是x和y进行比较,将较大的值赋给z.

当然这个例子感觉有些“为赋新词强说愁”的意味,但是有时候确实会存在参数自身发生变化导致宏结果的不确定性。所以在使用宏的时候尽量不要去传带有++、- -这些运算符的参数,很容易出现意想不到的结果。

5.宏替换的规则

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

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

注意:

  1. 宏参数和#define定义中可以出现其它#define定义的符号。但是宏,不能递归!
  2. 当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索。

举例说明第二点如下所示:

#include <stdio.h>
#define MAX(a,b) ((a)>(b)?(a):(b))
int main()
{
	int a = MAX(3, 4);
	printf("MAX(3,4)=%d\n",a);//字符串中的宏不会被替换
	return 0;
}

VS2022 X64输出结果如下

MAX(3,4)=4

6.宏和函数的对比

宏通常被应用于执行简单的运算,比如在两个数字=中找出较大的一个时,写成下面的宏,更有优势一些。

#define MAX(a,b) ((a)>(b)?(a):(b))

为什么不用函数来完成这个任务呢?
原因有二:

  1. 用于调用函数和从函数返回的代码可能比实际执行这个小型计算工作所需要的时间更多,所以宏比函数在程序的规模和速度方面更胜一筹
  2. 更为重要的是,函数的参数必须声明为特定的类型,所以函数只能在类型合适的表达式上使用,反之,这个宏可以适用于整型、长整型、浮点型等可以用>比较的类型,宏的参数与类型无关

和函数相比宏的劣势:

  1. 每次使用宏的时候,一份宏定义的代码将插入到程序中。除非宏比较短,否则可能大幅度增加程序的长度。
  2. 宏是没法调试的。
  3. 宏由于类型无关,也就不够严谨。
  4. 宏可能带来运算符优先级的问题,导致结果可能不确定。

宏有时候可以做函数做不到的事情,比如,宏的参数可以出现类型,但是函数不可以。

举例代码如下:

#include <stdio.h>
#include <stdlib.h>
#define MALLOC(num,type)\
	(type*)malloc(num*sizeof(type))
int main()
{
	int* p= MALLOC(10, int);
	if (p = NULL)
	{
		perror("MALLOC");
		return 1;
	}
	for (int i = 0; i < 10; i++)
		p[i] = i + 1;
	for (int i = 0; i < 10; i++)
		printf("%d ",p[i]);
	return 0;
}

综上,宏和函数的比较如下:
在这里插入图片描述

7.#和##运算符

7.1 #运算符

#运算符将宏的一个参数转换为字符串字面量,仅出现在带参数的宏替换列表中。
对于变量a,其值为10,我们想打印:the value of a is 10.
可以用以下宏定义:

#define PRINT(n) printf("the value of " #n " is %d\n",n)
//同一行多个字符串打印的时候会自动拼接,#n会自动转化为字符串

举例如下:

#include <stdio.h>
#define PRINT(n) printf("the value of " #n " is %d\n",n)
int main()
{
	int a=10,b=11,c=12;
	PRINT(a);
	PRINT(b);
	PRINT(c);
	return 0;
}

VS2022 X64输出结果如下所示
在这里插入图片描述

7.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 GENERIC_MAX(type)\
type type##_max(type x,type y)\
{\
	return x > y ? x : y;\
}
GENERIC_MAX(int)
GENERIC_MAX(float)
int main()
{
	int a = int_max(2, 3);
	printf("%d\n", a);
	float b = float_max(2, 3);
	printf("%f\n", b);
	return 0;
}

VS2022 X64输出结果如下所示

3
3.000000

在实际开发中##使用较少,很难举出非常贴切的例子。

8.命名约定

一般来讲,函数和宏的使用语法很相似,
命名习惯为:宏定义一般都会将name的所有字母设置为大写,函数名一般只有开头字母大写。

9.undef

该指令用于移除一个宏定义。

#define M 10
#undef M//首先移除旧的宏定义名,才能重新定义
#define M 10

10.命令行指令

许多C的编译器提供了一种能力,允许在命令行中定义符号,用于启动编译过程。
例如:可以用于同一个源文件编译一个程序的不同版本(假定某个程序中声明了一个某个长度的数组,如果机器内存有限,我们需要一个很小的数组,但是另外一个机器内存大一些,我们需要一个大一些的数组)。

#include <stdio.h>
int main()
{
 int array [ARRAY_SIZE];
 int i = 0;
 for(i = 0; i< ARRAY_SIZE; i ++)
 {
 array[i] = i;
 }
 for(i = 0; i< ARRAY_SIZE; i ++)
 {
 printf("%d " ,array[i]);
 }
 printf("\n" );
 return 0;
}

编译指令:

//linux环境
gcc -D ARRAY_SIZE=10 programe.c

11.条件编译

在编译一个程序的时候,可以使用条件语句选择性编译语句。
比如:调试性的代码。

#include <stdio.h>
#define __DEBUG__
int main()
{
 int i = 0;
 int arr[10] = {0};
 for(i=0; i<10; i++)
 {
 arr[i] = i;
 #ifdef __DEBUG__
 printf("%d\n", arr[i]);//为了观察数组是否赋值成功。 
 #endif //__DEBUG__
 }
 return 0;
}

常见的条件编译指令:
1.

#if 常量表达式
		//...
#endif
//常量表达式由预处理器求值

如:

#define __DEBUG__ 1
#if __DEBUG__ 
	printf("haha\n");
#endif

2.多个分支的条件编译

#if 常量表达式
		//...
#elif 常量表达式
		//...
#else
		//...
#endif

举例代码如下:

#include <stdio.h>
#define __DEBUG__ 2
#if __DEBUG__==0
	printf("hehe\n");
#elif __DEBUG__==1
	printf("haha\n");
#else 
	printf("heihei\n");
#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

12.头文件的包含

12.1 头文件包含方式

12.1.1 本地头文件包含

#include "filename"

查找策略:先在源文件所在目录下查找,如果该头文件未找到,编译器就像查找库函数头文件一样在标准位置查找文件。
Linux环境的标准头文件路径:

/usr/include

VS环境的标准头文件路径:

C:\Program Files (x86)\Microsoft Visual Studio 12.0\VC\include
//这是VS2013的默认路径

根据安装路径去找。

12.1.2 库文件包含

#include <filename.h>

一般标准头文件格式为#include <stdio.h>,即使用<>,而不是"";使用""的头文件查找策略如上所述,使用<>的头文件会直接在标准位置查找。所以,如果对库文件使用"",会先在目录文件下寻找,但这是浪费时间的,效率较低,并且也不容易区分到底是库文件还是本地文件

12.2 嵌套文件包含

#include指令可以使一个文件被编译,预处理会将这条指令的内容替换为该文件包含的内容。
但是,如果多次引用头文件,头文件内容就会被多次包含,对编译的压力比较大。
test.h

void test();
struct Stu
{
	int id;
	char name[20];
};

test.c

#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
int main()
{
	return 0;
}

如果是这样的话,test.c文件将5次包含test.h的内容,如果tets.h内容比较多,预处理代码剧增。如果工程较大,公共使用的头文件被多次引用,不做任何处理,导致预处理后的代码冗余。

每个头文件的开头写:

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

或者

#pragma once

可以避免头文件的重复引入
注:推荐《⾼质量C/C++编程指南》中附录的考试试卷(很重要)。
笔试题:

  1. 头⽂件中的 ifndef/define/endif是⼲什么用的?
  2. #include <filename.h> 和 #include “filename.h” 有什么区别?

13.其他预处理指令

#error
#pragma
#line
...//大家有兴趣自行了解
#pragma pack()//用于改变编译器结构体的默认对齐数

参考《C语言深度解剖》学习。


总结

“路漫漫其修远兮,吾将上下而求索。”是C语言最后一篇啦,下次就是数据结构啦!头文件、宏定义等内容比较多,记得消化哟~

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

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

相关文章

还不懂BIO,NIO,AIO吗

BIO&#xff08;Blocking I/O&#xff09;、NIO&#xff08;Non-blocking I/O&#xff09;和 AIO&#xff08;Asynchronous I/O&#xff09;是 Java 中三种不同的 I/O 模型&#xff0c;主要用于处理输入 / 输出操作。 一、BIO&#xff08;Blocking I/O&#xff09; 定义与工作原…

主窗口的设计与开发(二)

主窗口的设计与开发&#xff08;二&#xff09; 前言 在上一集当中&#xff0c;我们完成了主窗口的初始化&#xff0c;主窗口包括了左中右三个区域。我们还完成了对左窗口的初始化&#xff0c;左窗口包括了用户头像、会话标签页按钮、好友标签页按钮以及好友申请标签页按钮。对…

JavaFX基本控件-TextField

JavaFX基本控件-TextField 常用属性textpromptTextpaddingalignmentwidthheighttooltipbordereditabledisablevisible 格式化整形格式化 实现方式Java实现fxml实现 常用属性 text 设置文本内容 textField.setText("测试数据");promptText 设置文本字段的提示文本&am…

django ubuntu 踩坑集锦

目录 1 ubantu mysql查看表结构2 导入同级目录文件出现未解析引用错误3 第三方包——tinymce富文本编辑器4 verbose_name,verbose_name_plural5 搜索路径的添加6 auto_now_add 和 auto_now7 auth_user的表结构8 在 Django 中定义 ForeignKey 字段时&#xff0c;必须指定 on_del…

共享内存喜欢沙县小吃

旭日新摊子好耶&#xff01; 系统从0开始搭建过通信方案&#xff0c;本地通信方案的代码&#xff1a;System V IPC 里面有共享内存、消息队列、信号量 共享内存 原理 两个进程有自己的内存区域划分&#xff0c;共享内存被创建出的时候是归属操作系统的&#xff0c;还是通过…

STM32G474内部温度传感器的使用

目录 概述 1 认识STM32G474内部温度传感器 1.1 温度传感器概述 1.2 温度传感器实现原理 1.3 读取温度方法 1.4 ADC模块上温度sensor的位置框图 2 STM32Cube创建项目 2.1 配置参数 2.2 STM32Cube生成的软件架构 3 温度数据算法实现 3.1 算法介绍 3.2 源代码 概述…

【H2O2|全栈】关于HTML(1)认识HTML

HTML相关知识 目录 前言 准备工作 WEB前端是什么&#xff1f; HTML是什么&#xff1f; 如何运行HTML文件&#xff1f; 标签 概念 分类 双标签和单标签 行内标签和块标签 HTML文档结构 预告和回顾 UI设计相关 Markdown | Md文档相关 项目合作管理相关 后话 前…

顶踩Emlog插件源码

源码介绍 顶踩Emlog插件源码 前些天看到小刀娱乐网的文章页面有了一些变化&#xff0c;那就是增加了一个有价值/无价值的顶踩按钮。 样式也是非常的好看 再加上两个表情包是非常的有趣。 写到了Emlog系统&#xff0c;效果如上图。 如何使用&#xff1a; 需要在echo_log.…

Python 算法交易实验88 QTV200日常推进-关于继续前进的思考

说明 念念不忘,必有回响 最初的时候&#xff0c;完全不了解架构方面的东西。后来决定要搞好这一块的时候&#xff0c;也就是不断的琢磨&#xff0c;到现在4年的时间&#xff0c;改变已经非常大了。现在习以为常的&#xff0c;都是当初梦寐以求的&#xff0c;而且在可见的未来 &…

论文精读-Supervised Raw Video Denoising with a Benchmark Dataset on Dynamic Scenes

论文精读-Supervised Raw Video Denoising with a Benchmark Dataset on Dynamic Scenes 优势 1、构建了一个用于监督原始视频去噪的基准数据集。为了多次捕捉瞬间&#xff0c;我们手动为对象s创建运动。在高ISO模式下捕获每一时刻的噪声帧&#xff0c;并通过对多个噪声帧进行…

龙芯+FreeRTOS+LVGL实战笔记(新)——05部署主按钮

本专栏是笔者另一个专栏《龙芯+RT-Thread+LVGL实战笔记》的姊妹篇,主要的区别在于实时操作系统的不同,章节的安排和任务的推进保持一致,并对源码做了改进和优化,各位可以先到本人主页下去浏览另一专栏的博客列表(目前已撰写36篇,图1所示),再决定是否订阅。此外,也可以…

行空板上YOLO和Mediapipe图片物体检测的测试

Introduction 经过前面三篇教程帖子&#xff08;yolov8n在行空板上的运行&#xff08;中文&#xff09;&#xff0c;yolov10n在行空板上的运行&#xff08;中文&#xff09;&#xff0c;Mediapipe在行空板上的运行&#xff08;中文&#xff09;&#xff09;的介绍&#xff0c;…

Node.js学习记录(一)

目录 一、文件读取 readFile 二、写入文件 writeFile 三、动态路径 __dirname:表示当前文件所处的目录、path.join 四、获取路径文件名 path.basename 五、提取某文件中的css、JS、html 六、http 七、启动创建web服务器 服务器响应 八、将资源请求的 url 地址映射为文…

idea插件开发的第二天-写一个时间查看器

介绍 Demo说明 本文基于maven项目开发,idea版本为2022.3以上,jdk为1.8本文在Tools插件之上进行开发 Tools插件说明 Tools插件是一个Idea插件,此插件提供统一Spi规范,极大的降低了idea插件的开发难度,并提供开发者模块,可以极大的为开发者开发此插件提供便利Tools插件安装需…

Spark的Web界面

http://localhost:4040/jobs/ 在顶部导航栏上&#xff0c;可以点击以下选项来查看不同类型的Spark应用信息&#xff1a; Jobs - 此视图将列出所有已提交的作业&#xff0c;并提供每个作业的详细信息&#xff0c;如作业ID、名称、开始时间、结束时间等。Stages - 此视图可以查…

新160个crackme - 050-daxxor

运行分析 需要破解Name和Serial PE分析 C程序&#xff0c;32位&#xff0c;无壳 静态分析&动态调试 ida找到关键字符串&#xff0c;双击进入函数 通过静态分析发现&#xff1a;1、Name通过计算得到Name12、对Name1第3、5、6分别插入byte_401290、byte_401290、word_401292&…

Weibull概率分布纸(EXCEL VBA实现)

在学习Weibull分布理论的时候&#xff0c;希望有一张Weibull概率纸&#xff0c;用来学习图解法。但是在度娘上没有找到的Weibull概率纸的电子版。在书上看到的Weibull概率纸&#xff0c;只能复印下来使用。于是萌生了自己制作Weibull概率纸的想法&#xff0c;帮助自己更好地学习…

综合案例-数据可视化-折线图

一、json数据格式 1.1 json数据格式的定义与功能 json是一种轻量级的数据交互格式&#xff0c;可以按照json指定的格式去组织和封装数据&#xff0c;json数据格式本质上是一个带有特定格式的字符串。 功能&#xff1a;json就是一种在各个编程语言中流通的数据格式&#xff0…

全倒装COB超微小间距LED显示屏的工艺技术,相比SMD小间距有何优势

全倒装COB&#xff08;Chip On Board&#xff09;超微小间距LED显示屏&#xff0c;在工艺技术上的革新&#xff0c;相较于传统的SMD&#xff08;Surface Mount Device&#xff09;小间距LED显示屏&#xff0c;展现出了多方面的显著优势。 首先&#xff0c;全倒装技术极大地提升…

JAVAEE初阶第七节(下)——物理原理与TCP_IP

系列文章目录 JAVAEE初阶第七节&#xff08;下&#xff09;——物理原理与TCP_IP 文章目录 系列文章目录JAVAEE初阶第七节&#xff08;下&#xff09;——物理原理与TCP_IP 一.网络层重点协议 1. IP协议如何管理地址 1.1 解决IP地址不够用的问题 1.2 网段划分 1.3 特殊的IP…