C 语言笔记:预处理

一、C 语言编译过程

  1. 预处理 gcc -E hello.c -o hello.i
    将 .c 中的头文件展开、宏展开,生成的文件是 .i 文件。
  2. 编译 gcc -S hello.i -o hello.s
    将预处理之后的 .i 文件生成 .s 汇编文件
  3. 汇编 gcc -c hello.s -o hello.o
    将 .s 汇编文件编译生成 .o 目标文件
  4. 链接 gcc hello.o -o hello elf
    将 .o 文件链接成目标文件

二、include

#include<> // 用尖括号包含头文件,在系统指定的路径下找头文件。
#include"" // 用双引号包含头文件,先在当前目录下找头文件,若找不到,再到系统指定的路径下找。
其中 # 表示这是一个预处理指令
注意
(1)include 经常用来包含头文件,可以包含 .c 文件,但是不要包含 .c。因为 incude 包含的文件会在预编译被展开,如果一个 .c 被包含多次,展开多次,会导致函数重复定义。所以不要包含 .c 文件。
(2)预处理只是对 include 等预处理操作不会进行语法检查。这个阶段有语法错误也不会报错,第二个阶段(编译阶段)才进行语法检查。

三、定义宏用 define

  1. 不带参宏
    #define PI 3.14的意思是在预编译的时候如果代码中出现了 PI 就用 3.14 去替换。
    宏的好处:只要修改宏定义,其他地方在预编译的时候就会重新替换。(“一改全改”)
    注意:宏定义后边不要加分号。
    例子:
#include <stdio.h>

// 宏定义只要改变定义时的常量表达式,
// 则代码中只要使用这个宏定义的地方都会改变
#define PI 3.1415926
int main()
{
    printf("PI = %lf\n", PI);

    double d = PI;
    printf("d = %lf\n", d);
    return 0;
}

执行结果:

PI = 3.141593
d = 3.141593

宏定义的作用范围,从定义的地方到本文件末尾。如果想要在中途终止宏的定义范围,用 undef PI
2. 带参宏
#define S(a,b) a*b
注意:带参宏的形参 a 和 b 没有类型名。预处理时,用实参代替形参,其他字符保留,如 S(2,4)

#define S(a, b) a*b
int main()
{
    printf("S = %d\n", S(2, 4));
    printf("S = %d\n", S(2 + 8, 4));
    return 0;
}

执行结果:

S = 8
S = 34

带参宏类似一个简单函数,将函数的参数进行设置,就可以传递给对应的表达式。
S(2 + 8, 4) 返回的是34,是因为它代表的是 2 + 8 × 4 = 34 .
如果把宏定义改成 #define S(a, b) ((a)*(b))S(2 + 8, 4) 返回 40 . 计算过程为 ((2 + 8) × (4))
3. 带参宏和带参函数的区别
带参宏被调用多少次就会展开多少次,执行代码的时候没有函数调用的过程,不需要压栈弹栈。所以带参宏,是浪费了空间,因为被展开多次,节省时间。
带参函数,代码只有一份,存在代码段,调用的时候去代码段取指令,调用的时候要压栈弹栈。有个调用的过程。所以说,带参函数是浪费了时间,节省了空间。
带参宏的形参没有类型名,带参函数的形参有类型名。

四、选择性编译

  1. 第一种选择性编译
#ifdef AAA
    代码段一
#else
    代码段二
#endif

上面的意思是:如果当前 .c 的 ifdef 上边定义过 AAA ,就编译代码段一,否则编译代码段二。

选择性编译和 if else 语句的区别,if else 语句都会被编译,通过条件选择性执行代码,而选择性编译只有一块代码被编译。
例子:

#define AAA
int main()
{
    #ifdef AAA
        printf("hello world\n");
    #else
        print("hello beijing\n");
    #endif
return0;
}

执行结果:

hello world
  1. 第二种选择性编译
#ifndef AAA
    代码段一
#else
    代码段二
#endif

和第一种互补。
防止头文件重复包含
3. 第三种选择性编译

#if 表达式
    代码段一
#else
    代码段二
#endif

如果表达式为真,编译第一段代码,否则编译第二段代码。
选择性编译在预编译阶段进行。

这种形式一般用于注释代码

#if 0
    要注释的代码
#endif
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值