C语言预处理
预处理:对一些预处理命令进行执行的过程
预处理命令:头文件包含;宏定义等,用#开头的一些命令。
在C语言中预处理命令不会直接被编译,而是在编译这些预处理命令时,将这些预处理命令进行解析,然后将预处理结果替换成真正的C语言能编译的C语句。
在编译之前,预处理命令define会自动将程序中所有的Pi替换成3.14.在编译时程序中是不会在存在预处理命令的。
一)宏定义
1)不带参数的宏定义
如: #define PI 3.14
①可以用宏定义来定义一个常量来设置一个数组的长度
#define arrr_len 100 int a[arrr_len]
②宏定义只是简单的替换,而不做正确性的检查,如果有#define 3.l4中的‘1’写成‘l’也不会报错,他只是简单的替换
如:#define N 3; 他就会将N替换成“3;”包括分号,因为宏定义,会将符号常量的任何东西都将看成一个字符串并替换掉
③通常宏定义出现在程序的开头,这个宏定义的作用域是本源文件。
④如果想要解除宏定义的作用域,就在你需要解除的地方加上: #undef XXX
⑤宏定义可以在定义的时候就层层替换
如: #define PI 3.14
#define R 5;
#define L 2*PI*R
注意:在宏定义的时候只是将这个看成字符串,做简单的替换,而不会做运算。
⑥ 在程序中如果在字符串中出现与宏名一样的字符,不会做替换
如: #define L = 2*3.14
Printf(“L=%d”,L);
2)带参数的宏定义
(1)#define s(a,b) a*b
S(2,3)替换成 2*3
(2) #define PI 3.14
#define S(r) PI*(r)*(r)
程序中:S(2+3) 替换成 3.14*2*2
(3) #define PI 3.14
#define S(r) PI*r*r
则s(2+3)可以替换成: 3.14*2+3*2+3
(4) (4)#define PI 3.14
#define S(r) PI*(r)*(r)
则s(2+3)可以替换成: 3.14*(2+3)*(2+3)
(5)#define D “%d”
#define D1 “\n”
#define D2 D D D1
printf(D2,5,6); 等价于 printf(“%d” “%d” “\n”,5,6)等价printf(“%d%d\n”,5,6)
二) 头文件包含
#include
三) 条件编译
定义:在C源码中,基本上所有的代码都要参与编译的过程,编译也是耗费时间的过程,因此我们在编译大量的C源码时,可以根据需要对一些源码进行编译
#define 标识符 1
#ifdef 标识符
程序段1;
#else
程序段2;
#endif
(红色的部分是可选部分,是可以省略的)
作用:如果ifdef后面的标识符是被#define定义过,则程序段1将会被编译,否则程序段2将会被编译。
例子:
#include <stdio.h>
#define int_16
int main(int argc, char *argv[])
{
#ifdef int_32
printf("int is:32");
#else
printf("int is:16");
#endif
return 0;
}
注意:#define int_32这个宏,后面可以是任意的字符,符号。或者什么都没有。
#define 标识符
#ifndef 标识符
程序段1;
#else
程序段2;
#endif
(红色的部分是可选部分,是可以省略的)
(1)
#if 表常量达式
程序段1;
#else
程序段2;
#endif
(红色的部分是可选部分,是可以省略的)
(2)嵌套
#if 表常量达式1
程序段1;
#else
#if 表常量达式2
程序段2;
#else
程序段3;
#endif
#endif
(5)
#if 表常量达式1
语句段1;
#elif 表常量达式2
语句段2;
#elif 表常量达式3
语句段3
#else
语句段4
#endif
四)C语言的内嵌宏
_ _ func__当前语句所在源码的哪一个函数中
_ _ LINE__和_ _ FILE__包含正在编译的程序的行号和文件名。
_ _ DATE _ _和内容形如month/day/year(月/日/年)的串,代表源文件翻译成目标码的日期。
_ _ TIME _ _中的串代表源代码编译成目标码的时间,形如hour:minute:second(时:分:秒)
如果_ _ STDC__的内容是十进制常数1,则表示编译程序的实现符合标准C。
附:
在预处理中,我们经常用到define ,这里得提到 typedef ,二者用法很相似,大家还不要搞混;具体的区别是什么呢?
1、typedef 是C语言关键字
2、define 是预处理命令
这里介绍一下typedef的应用,typedef经常被用于为另一种数据类型定义一个新名字,在数据结构中更是频繁出现
typedef <已有数据类型> <新数据类型>
下面举个例子介绍:
typedef struct _node_
{
int data;
struct _node_ *next;
}listnode,*linklist;
这里定义了两个新的数据类型listnode和linklist。其中listnode等价于数据类型struct node 而linklist等价于stuct node *;
或者,我们这样理解,上面的定义就相当于:
struct _node_
{
int data;
struct _node_ *next;
};
typedef struct _node_ listnode;
typedef struct _node_ *linklist;