C语言学习笔记之预处理命令

8.1预处理命令

编译是针对单个源文件的,一次编译操作只能编译一个源文件,如果程序中有多个源文件则需要多次编译操作

预处理命令需要放在所有函数之外。
C语言提供了多种预处理功能:

  1. 宏定义
  2. 文件包含
  3. 条件编译

作用:
便于调试、修改、阅读和移植,利于模块化设计。
预处理的工作:
把代码当成普通文本,根据设定的条件进行文本替换,将替换以后的结果交给编译器。

8.2#include的用法

#include称为文件包含命令,用来引入头文件。
#include是C语言预处理命令的一种

#include后面跟<>与“”的辨析
<>会指定编译器到系统目录下查找头文件
“”会指定编译器先到当前目录查找,一般用于自定义的头文件,如果找不到再到系统目录下查找头文件

(如果“”当前目录文件与系统目录文件重名有什么影响?)

习惯使用尖括号来引入标准头文件,使用双引号来引入自定义头文件(,这样一眼就能看出头文件的区别。

注:无论是标准头文件还是自定义头文件,都只能包含变量和函数声明,不能包含函数定义,否则会在多次引入时出现重复定义的错误。

8.3C语言宏定义

#define 宏名 字符串
例如
#define N 100

注:
#define不是说明或语句,不用加;
#define可以用#undef取消定义
#define必须在函数之外
#define的宏名被引号包围则失效
#define允许多层嵌套
#define UINT unsigned int宏定义数据类型

#define与typedef的辨析:
#define只是简单的字符串替换,由预处理器处理。
typedef在编译阶段由编译器处理,相当于给数据类型起新名字(新的数据类型)

8.4C语言带参数的宏定义

#include<stdio.h>
#define MAX(a,b) a>b?a:b
int main() {
    
    

	int x,y,max;
	printf("input two number");
	scanf_s("%d%d",&x,&y);
	max = MAX(x,y);
	printf("%d\n",max);
	return 0;
}

在带参宏定义中,不会为形式参数分配内存,因此无需指明数据类型
在带参宏调用中,实参包含了具体的数据,因此必须指明数据类型

辨析:
函数中,形参和实参是两个不同的变量,都有自己的作用域,调用时会把实参的值传递给形参;
在带参数的宏中,只是符号的替换,不存在值传递的过程。

#include<stdio.h>
#define S(y) (y)*(y)
int main() {
    
    
	int a,v;
	printf("input a number:");
	scanf_s("%d",&a);
	v = S(a+1);
	printf("%d\n",v);
	return 0;
}
#include<stdio.h>
#define S(y) y*y
int main() {
    
    
	int a,v;
	printf("input a number:");
	scanf_s("%d",&a);
	v = S(a+1);
	printf("%d\n",v);
	return 0;
}

8.5带参宏定义和函数的区别?

宏定义仅仅是字符串的替换,不会对表达式进行计算;
宏在编译之前处理,不参加编译、不占用内存;
函数是一段可以重复使用的代码、会被编译、会占内存,每次调用函数就是执行这块内存中的代码。

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

8.6宏参数的字符串化和宏参数的连接

#define 宏名 宏参数
#用在宏参数前可以将宏参数转换为字符串

#include<stdio.h>
#define STR(s) #s

int main() {
    
    

	printf("%s\n",STR(HAHAHA));
	printf("%s\n",STR("HAHAHA"));
	return 0;
}

##用来连接字符串

8.7C语言中几个预定义的宏

#include<stdio.h>
#include<stdlib.h>
int main() {
    
    
	printf("%s\n", __DATE__);
	printf("%s\n", __TIME__);
	printf("%s\n", __FILE__);
	printf("%s\n", __LINE__);
	system("pause");
	return 0;
}

8.8C语言条件编译

条件编译:这种能够根据不同情况编译不同代码,产生不同目标文件的机制称为条件编译。
条件编译需要多个预处理命令:
#if

#elif

#else

#endif

#ifdef的用法:

#ifdef

#else

#endif

debug和release的辨析:
debug用于调试
release用于发布版本,提高了运行效率

#include<stdio.h>
#include<stdlib.h>

int main() {
    
    
#ifdef _DEBUG
	printf("debug");
	
#else
	printf("not debug");
#endif // DEBUG

	system("pause");
	return 0;
}

#ifndef的使用:
#ifndef

#else

#endif

#if #ifdef #idndef的辨析:
#if 整型常量表达式
#ifdef 宏名
#ifndef 宏名

#ifdef 宏名:

#include <stdio.h>
#define NUM 100
int main() {
    
    
#ifdef NUM****
	printf("NUM: %d\n", NUM);
#else
	printf("NUM Error\n");
#endif
	return 0;
}
#include <stdio.h>
#define NUM 100
int main() {
    
    
#ifdef NU****M****
	printf("NUM: %d\n", NUM);
#else
	printf("NUM Error\n");
#endif
	return 0;
}
#include<stdio.h>

#define M 100
#define N 200

int main() {
    
    
#if (defined M &&defined N)
	printf("right\n");
#else
	printf("wrong\n");
#endif // defined M=100 &&defined N=200
	system("pause");
		return 0;
}
#include<stdio.h>
#define M 100
#define N 200
int main() {
    
    
#if (defined M && defined Q)
	printf("right\n");
#else
	printf("wrong\n");
#endif // defined M=100 &&defined N=200
	system("pause");
		return 0;
}

8.9#error命令,阻止程序编译

8.10C语言预处理命令小结

预处理命令,在编译器进行编译之前对源代码做某些转换

预处理功能是C语言特有的功能,它是在对源程序正式编译前由预处理程序完成的,程序员在程序中用预处理命令来调用这些功能。

宏定义可以带有参数,宏调用时是以实参代换形参,而不是“值传送”。

为了避免宏代换时发生错误,宏定义中的字符串应加括号,字符串中出现的形式参数两边也应加括号。

文件包含是预处理的一个重要功能,它可用来把多个源文件连接成一个源文件进行编译,结果将生成一个目标文件。

条件编译允许只编译源程序中满足条件的程序段,使生成的目标程序较短,从而减少了内存的开销并提高了程序的效率。

使用预处理功能便于程序的修改、阅读、移植和调试,也便于实现模块化程序设计。

猜你喜欢

转载自blog.csdn.net/BJTUYBYUAN/article/details/115186615