C语言最终讲:预处理详解

C语言最终讲:预处理详解

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

结语:这一讲是C语言基础知识的最后一讲了,后续将会学习数据结构相关的知识,坚持不易,希望各位都能坚持在自己所干的事情上,我们共勉

这一讲讲的是预处理相关的内容,上一讲虽然我们已经了解了一些关于#define相关的知识,这一讲我们讲详细阐述它的作用和缺陷

1.预定义符号

C语言中有着一些预定义符号,可以直接使用,是在预处理阶段进行处理的:

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

它们的使用方法为:

int main()
{
	printf("%s\n", __FILE__);//打印当前文件所在的位置
	printf("%d\n", __LINE__);//打印行号
	printf("%s\n", __DATE__);//打印日期
	printf("%s\n", __TIME__);//打印时间
	//printf("%d\n", __STDC__);//err,表明VS并不完全遵循ANSI C标准

	return 0;
}

运行结果如下:
在这里插入图片描述

2.#define定义常量

我们知道,define定义的常量是直接替换的,所以有着下面的几处用法:

#define TEST 20 //方法1,直接定义常量
#define I int//方法2:可以为关键字替换成一个更简便的名字
#define CASE break;case//方法3:这是一种很“奇葩”的用法,建议还是别用了

int main()
{
	//使用起来也很方便,方法1:
	int a = TEST;//直接使用即可

	//方法2:
	I b = 20;
	printf("%d\n", b);

	//方法3:
	int input = 1;
	scanf("%d", &input);
	switch (input)
	{
	case 1:
	CASE 2 ://这样后面就不用加break了
	CASE 3 :
		break;
	}
	return 0;
}

对于加;问题,解答如下:

#define ROW 20;//假设#define定义时加上了;

int main()
{
	int a = ROW;//这是a就会被替换成int a = 20;;这是后面就会有两个;,特殊情况下会出现错误
	//总结:加上;可能会出现错误,不加;肯定不会出现错误,所以还是不要加上;

	return 0;
}

3.#define定义宏

定义方式:

#define name( parament-list ) stuff
name为宏名称
parament-list是由逗号隔开的符号表,可能出现在stuff中
stuff可以是一个计算方式,也可以是一个指令等等
//需要注意的是,符号表的左括号必须要和name紧密相连,否则括号里的内容也会被当成是stuff中的一部分

使用实例:

#define MUL(x) x*x//定义了一个宏,实现x的平方

int main()
{
	int a = 5;
	int b = MUL(a);

	printf("b = %d\n", b);
	//结果为25
	return 0;
}

看起来使用起来很好用,但是我们也要注意符号的优先级问题,如下:

#define ADD(x) x+x

int main()
{
	int a = 5;
	int b = ADD(a) * ADD(a);
	printf("%d\n", b);
	//我们可能会误以为结果为10 * 10,这不就是100吗,但是并非如此
	//b会被替换成int b = 5+ 5*5 +5,根据符号优先级问题,结果为35
	return 0;
}

知道错误就要改正,所以改正方法如下:

#define ADD(x) ((x)+(x)) //在x上加一个括号,因为x为表达式,如果传入5+5,也要被括起来
               //总体上再加上一个括号,是为了将它们算成一个整体

int main()
{
	int a = 5;
	int b = ADD(a) * ADD(a);
	printf("%d\n", b);
	//这样才能计算出一个整体的值,此时结果为100
	return 0;
}

所以,对于宏的定义,我们不能吝惜括号

4.带有副作用的宏参数

什么是带有副作用呢,其实就是表达式求值会改变原来的参数的值,如下:

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

为了验证此写法的危害性,我们举例来说明:

#define MAX(a, b) ( (a) > (b) ? (a) : (b) )
int main()
{
	int x = 5;
	int y = 8;
	int z = MAX(x++, y++);
	//这里会被替换成((x++)>(y++) ? (x++):(y++))
	//先看问号左边,后置++,先使用后++,5>8为假,所以看的是y++的值,还是先使用后++,但是要注意的是
	//由于++,此时y的值已经变成了9,所以z被赋值为9,y再++,值位10
	//而x只++了一次,因为第二次只对y进行了++,所以x的值为6

	printf("x=%d y=%d z=%d\n", x, y, z);//输出的结果是什么?
	//所以输出的结果应该为6 10 9
	return 0;
}

5.宏替换的规则

1.在调用宏时,首先看参数中是否包含由#define定义的符号,如果有,它们首先被替换:

#define R 2
#define RR R+2
//其中RR中包含了一个常量的定义R,所以R先被替换成2,再计算RR

2.替换文本随后会被插入到原来文本所在的位置
3.最后还会对文件进行扫描,看是否仍然存在#define定义的符号,如果有,继续进行上面的两个步骤

对于宏,需要注意的是,它并不能出现递归

6.宏和函数的对比

上面我们了解了宏,那么既然宏这么好用,那我们是不是就能够将宏代替为函数使用呢,肯定是不能的,下面我们来分析宏相较于函数的利与弊

6.1宏的优势

1.对于简单的任务,宏有着天然的优势:

#define MAX(a, b) ((a)>(b)?(a):(b))
1.首先,宏的参数是类型无关的,也就是说,宏参数能够接受任意类型的值,在特定条件下是很好用的
2.其次,宏在使用过程中也是一个简单的替换,函数不同,函数在使用前后会产生空间的创建和销毁,使用效率较慢

2.相较于函数,宏有着函数不能够实现的情况:

//如果我们要使用函数进行空间的开辟:
int* Mal(int a, int size)
{
	//我们首先会将返回值和参数确定,也就是说,它们的类型都是固定的,用于整形开辟的函数就不能用于浮点型类型的开辟
	return (int*)malloc(a * sizeof(int));
}

//但是当我们使用宏时:
#define MAL(num, type)  \
        (type*)malloc(num * sizeof(type))
int main()
{
	//使用函数开辟空间
	int* pa = Mal(10, sizeof(int));//假设我们要开辟10个整形的空间

	//使用宏开辟空间
	int* pb = MAL(10, int);
	//1.我们可以将int直接作为参数传入
	//2.此时宏的使用包含的情况更多,更好用
	return 0;
}

6.1.1\符号

上面我们使用了一个\符号,我们看看这是怎么个事:

#define MAL(num, type)  \
        (type*)malloc(num * sizeof(type))
//这个宏定义中使用了\符号,它是一个换行符,当宏里的内容过多时,就可以在后边加上一个\符号进行换行
//但是要注意的是,\后边什么也不能有,包括空格
//可以理解为:一个\抵消了一个\n,所以后边不能跟任何东西

6.2宏的劣势

1.每次使用宏时,一份宏代码就会被插入到程序中,如果宏很长的话,会大幅增加程序的长度
2.宏在预处理阶段就被替换,不能够被调试
3.宏由于类型无关,所以也容易出现问题
4.宏可能造成运算符优先级的问题

对于宏和函数的一个对比,我们可以看图:
图不是很好,大家对付着看
在这里插入图片描述

7.#和##

7.1#运算符

#所实现的操作被称为“字符串化”,作用是将宏的一个参数转换成字符串变量:

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

int main()
{
	int a = 10;

	PRINT(a);
	//此时会打印出:The value of a is 10
	//预处理阶段,代码会被替换成:printf("the value of ""a" " is %d", a);
	//也就是说,#n就表示"n",对于printf,多个""会被合并成一个
	return 0;
}

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 MAX(type)                      \
		type type##_max(type x, type y)\
		{                              \
			return x > y ? x : y;      \
		}

MAX(int);//这时我们可以依靠宏来创建一个函数,函数名为int_max

int main()
{
	//使用方法也比较简单:
	int ret1 = int_max(2, 3);//我们直接使用创建好的函数即可
	printf("%d\n", ret1);
	return 0;
}

8.命名约定

在命名宏和函数时,有着一个不成文的约定:

1.把宏名全部大写
2.函数名不要全部大写

9.#undef

用于移除一个宏定义,使用方法如下:

#define A 20

int main()
{
	int a = A;
	printf("%d\n", a);//20

#undef A
	int b = A;//err
	return 0;
}

10.命令行定义

在不同的场景下,我们定义的数组长度需求可能不同,所以许多C的编译器提供了一种命令行定义的能力,允许在命令行中,也就是使用前对一些变量进行赋值

例子:

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
这里表示给ARRAY_SIZE赋值为10,然后再执行程序

11.条件编译

对于一些为了调试而使用的代码,删除了可惜,保留了又费事,所以我们可以通过条件编译来选择性编译

常见的条件编译指令:

1.
#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

总结:

1.#if#endif是一体的,它们两个必须同时使用
2.#if#elif就相当于if else
3.#if defined(symbol)表示如果定义了symbol,也可以写成:#ifdef symbol
4.#if !defined(symbol)表示如果没有定义symbol,也可以写成:#ifndef symbol

它的原理为:

//使用原理为:
//当条件为真时,保留代码
//条件为假时,删除代码
int main()
{
#if 1
	printf("haha\n");//条件为真,保留代码
#elif 0
	printf("hehe\n");//条件为假,删除代码
#endif
	return 0;
}
//所以上面的代码在预处理之后就变成了:
int main()
{
	printf("haha\n");//条件为真,保留代码
	return 0;
}

12.头文件的包含

头文件有两种包含方式:本地文件包含和库文件包含,它们的包含方式是什么呢,我们来看:

12.1本地文件包含

#include "filename"
使用“”来包含

查找策略:

1.现在源文件目录下查找,也就是这个路径:

在这里插入图片描述

2.如果没有找到,在库函数头文件中进行查找
3.如果找不到,提示编译错误

Linux环境下标准头文件的路径:

/usr/include

VS环境下(现在VS环境下头文件的路径应该会有差异,这里只当成一个代表):

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

12.2库文件包含

直接去标准路径下查找,如果没有找到就提示编译错误

那么我们就要问了,是不是库文件的包含也可以使用""呢,原理上来说可以,但是缺点也很明显:
1.效率低
2.这样就不能够分辨出哪个是包含的是标准库文件,哪个是包含本地文件

12.3嵌套文件包含

当我们使用#include包含多个头文件时,会发生什么呢:

在这里插入图片描述
在预处理阶段,竟然会有5个头文件被写入了.c文件中,因为头文件的包含其实就是将头文件的内容写入.c文件中,情况如下:

struct Stu
{
	int id;
	char name[20];
};
struct Stu
{
	int id;
	char name[20];
};
struct Stu
{
	int id;
	char name[20];
};
struct Stu
{
	int id;
	char name[20];
};
struct Stu
{
	int id;
	char name[20];
};

#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
int main()
{

	return 0;
}

为了避免这种情况,所以我们用到了:条件编译,方法如下:

//头文件的改进如下:
//方法1:条件编译
#ifndef __TEST_H__
#define __TEST_H__//也就是说,如果没有__TEST_H__的话,定义一个__TEST_H__,这样后续就有了__TEST_H__,就不会再包含这个头文件了
struct Stu
{
	int id;
	char name[20];
};
#endif

//方法2:
#pragma once
//这是一个预处理指令,用于告诉编译器该头文件之应该被包含一次,通常推荐放在头文件开头

13.其他预处理指令

#error
#pragma
#line
...
不做介绍,⾃⼰去了解

可以参考《C语言深度解剖》一书进行学习

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

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

相关文章

OpenAI 推出适用于 .NET 的 OpenAI 库

OpenAI 推出适用于 .NET 的 OpenAI 库 微软最近宣布推出面向.NET开发人员的官方OpenAI库&#xff0c;该库支持OpenAI的全套API和最新的GPT-4o旗舰模型。这个模型可以实时进行音频、视觉和文本推理。 OpenAI .NET API库 目前&#xff0c;微软已经发布了OpenAI .NET API库的第一…

树二叉树

树 ​ 树是 n&#xff08;n≥0&#xff09;个结点的有限集。当 n 0时&#xff0c;称为空树。在任意一颗非空树中应满足&#xff1a; &#xff08;1&#xff09;有且仅有一个特定的称为根的结点。 &#xff08;2&#xff09;当 n > 1时&#xff0c;其余结点可分为 m&…

bitset用法

参考:https://blog.csdn.net/weixin_45697774/article/details/105563993 题目:https://leetcode.cn/problems/maximum-total-reward-using-operations-ii/description/ class Solution { public:int maxTotalReward(vector<int>& rewardValues) {bitset<10000…

未卸载干净的proteus安装教程7.8

提醒&#xff1a; 针对第一次安装推荐博文&#xff1a;https://jingyan.baidu.com/article/656db918f8590de381249cbf.html 1、一定要以管理员身份运行软件。 2、以管理员身份运行软件后&#xff0c;默认的ISIS Professional路径是C:\Program Files \Labcenter Electronics\…

写给大数据开发,如何去掌握数据分析

这篇文章源于自己一个大数据开发&#xff0c;天天要做分析的事情&#xff0c;发现数据分析实在高大上很多&#xff0c;写代码和做汇报可真比不了。。。。 文章目录 1. 引言2. 数据分析的重要性2.1 技能对比2.2 业务理解的差距 3. 提升数据分析能力的方向4. 数据分析的系统过程4…

初识springclould到生产者消费者的RPC通信

SpringClould SpringBoot和SpringClould搭建springcloud创建项目管理实体类模块服务提供者模块消费者 Eureka 服务注册与发现 SpringBoot和SpringClould springboot和springclould都是spring系列的衍生品&#xff0c;都可以在spring的官网找到对应的参考文档和学习路线以及核心…

PHP“well”运动健身APP-计算机毕业设计源码87702

【摘要】 随着互联网的趋势的到来&#xff0c;各行各业都在考虑利用互联网将自己的信息推广出去&#xff0c;最好方式就是建立自己的平台信息&#xff0c;并对其进行管理&#xff0c;随着现在智能手机的普及&#xff0c;人们对于智能手机里面的应用“well”运动健身app也在不断…

安装systemd-bootchart

要安装systemd-bootchart&#xff0c;你可以按照以下步骤进行&#xff1a; 步骤一&#xff1a;更新软件包列表 首先&#xff0c;打开终端并更新你的软件包列表&#xff0c;以确保你拥有最新的可用软件包信息。运行以下命令&#xff1a; sudo apt update步骤二&#xff1a;安…

Python 连接 MySQL 及 SQL增删改查(主要使用sqlalchemy)

目录 一、环境 二、MySQL的连接和使用 2.1方式一&#xff1a;sql为主 2.1.1创建连接 2.1.2 表结构 2.1.3 新增数据 ​编辑 2.1.4 查看数据 ​编辑 2.1.5 修改数据 2.1.6 删除数据 2.2方式二&#xff1a;orm对象关系映射 2.2.1 mysql连接 2.2.2 创建表 2.2.3 新增…

ORA-01652 表空间不够解决方案

前章&#xff1a;出现表空间不足不要手动强制删除对应数据文件存储目录下的DBF文件&#xff0c;需要用SQL语句进行数据文件的DROP&#xff0c;否则会导致ORA-01033报错&#xff0c;因为我没有开启数据库的归档所以不能通过RECOVER的形式找回数据文件最后只能重装本地ORACLE。 …

大模型如何通过token进行推理?

大型模型通过token进行推理的过程通常涉及以下步骤&#xff1a; 1、Tokenization&#xff08;分词&#xff09;&#xff1a;首先&#xff0c;输入文本或序列被分割成tokens&#xff0c;这些tokens通常是单词、子词或字符的序列。这一步通常由预训练模型的tokenizers完成。 2、…

转型AI产品经理(7):“格式塔原则”如何应用在Chatbot产品中

格式塔原则&#xff0c;又称为完形原则&#xff0c;它是一组关于人类如何感知视觉元素的心理学理论&#xff0c;这些原则说明了大脑如何将分散的视觉元素整合为有意义的整体&#xff0c;即使这些元素本身可能是分离的&#xff0c;帮助我们理解人们如何组织和解释复杂的视觉信息…

2013年 阿拉斯加巴罗活动层厚度和土壤含水量

Pre-ABoVE: Active Layer Thickness and Soil Water Content, Barrow, Alaska, 2013 ABoVE前&#xff1a;阿拉斯加巴罗活动层厚度和土壤含水量&#xff0c;2013年 简介 文件修订日期&#xff1a;2018-01-10 数据集版本&#xff1a;1 摘要 该数据集提供了 2013 年 8 月在…

速卖通如何放关联?

大家都知道&#xff0c;想要进行多账号操作必须一再小心&#xff0c;否则会有很大的关联风险&#xff0c;而账号关联所带来的后果是卖家绝对不能轻视的&#xff0c;严重的话会导致封号&#xff0c;这样一来自己前期的辛苦运营就全都打水漂了&#xff0c;因此防关联很重要&#…

在idea中创建Scala项目教程

1.下载Scala支持插件 文件-设置-插件-marketplace 搜索Scala 下载 2.创建项目 文件-新建-项目-新项目-构建系统maven 3.创建Scala目录 Scr-main(右键)-新建-目录&#xff08;Scala回车键&#xff09;-scala(右键)-将项目标记为-源代码根目录 4.对当前项目引入Scala支持 未添…

C++:day5

思维导图 例题 #include <iostream> using namespace std; class RMB { private:int yuan;int jiao;int fen;static int count;public:RMB(){count;}RMB(int yuan, int jiao, int fen) : yuan(yuan), jiao(jiao), fen(fen){count;}const RMB operator(const RMB &R)…

14.《C语言》——【牛客网BC116—BC123题目讲解】

亲爱的读者&#xff0c;大家好&#xff01;我是一名正在学习编程的高校生。在这个博客里&#xff0c;我将和大家一起探讨编程技巧、分享实用工具&#xff0c;并交流学习心得。希望通过我的博客&#xff0c;你能学到有用的知识&#xff0c;提高自己的技能&#xff0c;成为一名优…

二叉树最大宽度

文章目录 前言二叉树最大宽度1.题目解析2.算法原理3.代码编写 总结 前言 二叉树最大宽度 1.题目解析 给你一棵二叉树的根节点 root &#xff0c;返回树的 最大宽度 。 树的 最大宽度 是所有层中最大的 宽度 。 每一层的 宽度 被定义为该层最左和最右的非空节点&#xff08;即…

这个国际档案日,大比武放榜、直播预约、课件下载,一样都不能少!

关注我们 - 数字罗塞塔计划 - 2024年6月9日第十七个国际档案日来临&#xff0c;数字罗塞塔计划放大招&#xff1a;第二届大比武活动榜单揭晓、ARCHE-2024上海智慧档案高峰论坛直播预约、2024上半年度课件大礼包下载。如此大礼&#xff0c;岂能错过&#xff1f; PART.01 榜单…

SpringCloud-面试篇(二十四)

&#xff08;1&#xff09;Nacos如何支撑数十万服务注册的压力 小型企业来讲nacos压力没有那么大&#xff0c;但是想阿里&#xff0c;服务的数量可能会达到数万&#xff0c;那麽多的服务。当服务原来越多时&#xff0c;除了服务注册以外&#xff0c;还有服务的定时更新&#x…