一文带你搞懂C语言预处理宏定义

 更新时间:2022年10月12日 14:46:14   作者:期邈云汉  
这篇文章主要为大家详细介绍了C语言预处理宏定义#define,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

预定义符号

这些预定义符号都是语言内置的

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

VS环境下未定义__STDC__ ,说明Visual Studio并未完全遵循ANSI C。

#define

#define 定义标识符

#define name stuff  //名称;内容
#define MAX 1000
#define reg register          //为 register这个关键字,创建一个简短的名字
#define do_forever for(;;)     //用更形象的符号来替换一种实现
#define CASE break;case        //在写case语句的时候自动把 break写上。
// 如果定义的 stuff过长,可以分成几行写,除了最后一行外,每行的后面都加一个反斜杠(续行符)。
#define DEBUG_PRINT printf("file:%s\tline:%d\t \
                          date:%s\ttime:%s\n" ,\
__FILE__,__LINE__ ,       \
__DATE__,__TIME__ )


int main() {
	printf("%d\n", MAX);
	printf("%s\n", STR);
	do_forever;  //执行到这里会死循环
	return 0;
}

在define进行定义时,最好不要在后面加上分号 ;,替换时也会将分号替换,容易导致问题。

#define 1000;
int max = 0;
	if (3 > 1)
		max = MAX;
	else
		max = 0;
//报错

#define 定义宏

#define 机制包括了一个规定,允许把参数替换到文本中,这种实现通常称为宏(macro)或定义宏(define macro)

#define name( parament-list ) stuff

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

//int Max(int x, int y) {
//	return (x > y ? x : y);
//}
//若采用宏定义
#define Max(x,y) (x>y?x:y)
int main() {
	int a = 10;
	int b = 5;
	int max = Max(a, b);
	printf("%d\n", max);
	return 0;
}

最终得到较大值10。找到了宏的标识,直接进行符号替换。

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

例如:

#define SQUARE(x) x*x

int main() {
	printf("%d\n", SQUARE(5));
	printf("%d\n", SQUARE(5 + 1));  //宏的参数是完全替换的,相当于 5 + 1 * 5 + 1  结果为11 
	return 0;
}

做改进:

#define SQUARE(x) (x)*(x)  //这样就不容易出错
//最好最外层也加上括号

考虑如下计算:

#define Double(x) (x)+(x)
int main() {
	int a = 5;
	printf("%d\n", (10 * Double(a)));  //这里相当于 10 * (5) + (5)  结果为55
	return 0;
}

在宏替换内容上加上括号以保证得到预期的结果。

#define Double(x) ((x)+(x))

所以用于对数值表达式进行求值的宏定义都应该用这种方式加上括号,避免在使用宏时由于参数中的操作符或邻近操作符之间不可预料的相互作用。

替换规则

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

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

注意:

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

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

# 和##

如何把参数插入到字符串中?

#define PRINT(FORMAT, VALUE) printf("the value of " #VALUE " is "FORMAT "\n", VALUE)
 
//这里#可以把其后面的参数替换为字符串嵌入

int main() {
	printf("hello world\n");
	printf("hello ""world\n");  //天然合为一个字符串(发现字符串是有自动连接的特点的)
	int a = 100;
	printf("The value a is %d\n", a);
	int b = 20;
	printf("The value b is %d\n", b);

	//那么考虑能否将同一个功能的打印操作合并
	PRINT("%d", a);  //这里只有当字符串作为宏参数的时候才可以把字符串放在字符串中
	/*代码中的 #VALUE 会预处理器处理为:
		"VALUE" .*/
		
	return 0;
}

把宏对应的参数替换为字符串

##的作用

将分离的片段合成为一个符号

#define CAT(A,B) A##B
//把A和B组合成一个符号
int main() {
	int Windows11 = 2022;

	printf("%d\n", CAT(Windows, 11));

}

带副作用的宏参数

当宏参数在宏的定义中出现超过一次的时候,如果参数带有副作用,那么你在使用这个宏的时候就可能出现危险,导致不可预测的后果。副作用就是表达式求值的时候出现的永久性效果。

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

考虑宏:

//考虑宏的副作用
#define MAX(a, b) ( (a) > (b) ? (a) : (b) )
int main() {
	int x = 5;
	int y = 8;
	int z = MAX(x++, y++);
	printf("x=%d y=%d z=%d\n", x, y, z);
	//z = ((x++) > (y++) ? (x++) : (y++));  x = 6  y = 10  z = 9  
}

到此这篇关于一文带你搞懂C语言预处理宏定义的文章就介绍到这了,更多相关C语言预处理宏定义内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • C语言进阶数据的存储机制完整版

    C语言进阶数据的存储机制完整版

    这篇文章主要为大家完整的介绍了C语言进阶数据的存储机制,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步早日升职加薪
    2022-02-02
  • C语言的strcpy函数你了解吗

    C语言的strcpy函数你了解吗

    这篇文章主要为大家详细介绍了C语言的strcpy函数,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-03-03
  • C语言实现2048游戏

    C语言实现2048游戏

    这篇文章主要为大家详细介绍了C语言实现2048小游戏,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-07-07
  • 详解C语言实现推箱子的基本功能(2)

    详解C语言实现推箱子的基本功能(2)

    这篇文章主要为大家详细介绍了C语言实现推箱子的基本功能的方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-02-02
  • 你知道C语言函数调用常用的2种方式吗

    你知道C语言函数调用常用的2种方式吗

    本篇博客会讲解C语言函数调用的2种方式,分别是:传值调用和传址调用。这2种函数调用方式有什么区别呢?为什么会有不同的效果呢?分别有哪些用途呢?下面就来一一展开
    2023-04-04
  • OpenMP Parallel Construct的实现原理详解

    OpenMP Parallel Construct的实现原理详解

    在本篇文章当中我们将主要分析 OpenMP 当中的 parallel construct 具体时如何实现的,以及这个 construct 调用了哪些运行时库函数,并且详细分析这期间的参数传递,需要的可以参考一下
    2023-01-01
  • C++实现当前时间动态显示的方法

    C++实现当前时间动态显示的方法

    这篇文章主要介绍了C++实现当前时间动态显示的方法,涉及C++时间操作及Sleep方法的使用,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-07-07
  • C语言文件操作详解以及详细步骤

    C语言文件操作详解以及详细步骤

    文件(file)一般指存储在外部介质上数据的集合,比如我们经常使用的.txt, .bmp, jpg. .exe,.rmvb等等,下面这篇文章主要给大家介绍了关于C语言文件操作的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-06-06
  • C语言简析指针用途

    C语言简析指针用途

    C语言这门课程在计算机的基础教学中一直占有比较重要的地位,然而要想突破C语言的学习,对指针的掌握是非常重要的,本文将具体针对指针的基础做详尽的介绍
    2022-07-07
  • FFmpeg实战之利用ffplay实现自定义输入流播放

    FFmpeg实战之利用ffplay实现自定义输入流播放

    ffplay是FFmpeg提供的一个极为简单的音视频媒体播放器,可以用于音视频播放、可视化分析。本文将利用ffplay实现自定义输入流播放,需要的可以参考一下
    2022-12-12

最新评论