一、由源码到可执行程序的过程

1. 预处理: 源码经过预处理器的预处理变成预处理过的.i中间文件

1 gcc -E test.c -o test.i

2. 编译: 中间文件经过编译器编译形成.s的汇编文件

1 gcc -S test.i -o test.s

3. 汇编: 汇编文件经过汇编器生成目标文件.o(机器语言)

1 gcc -c test.s -o test.o

4. 链接: 链接器将目标文件链接成.exe可执行程序(Linux下是.elf)

1 gcc test.o -o test.exe

在整个过程中,预处理用预处理器,编译用编译器,汇编用汇编器,链接用链接器,这几个工具再加上其他额外的可能会用到的工具,合起来叫编译工具链。gcc/g++就是一个编译工具链,在实际工程中并不会去手动生成那么多中间文件,而是直接一步到位:

1 gcc test.c -o test.exe

其中,编译器的主要目的是编译源代码,即将.c的源代码(.i本质上就是预处理过的.c)转化成.s汇编代码。为了让编译器聚焦核心功能,就将一些非核心功能剥离到预处理器去了,也就是所让预处理器帮编译器做一些编译前的准备工作。

二、编程中常见的预处理

  1. 头文件包含:#include
  2. 注释
  3. 条件编译:#if #elif #endif...
  4. 宏定义

2.1 头文件包含

2.1.1 #include <>和 #include ""的区别

  • #include <>专门用来包含系统提供的头文件,如果使用<>,编译器只会到系统指定目录(编译器中配置的或OS配置的目录,如在Ubuntu中是/usr/include,编译器还允许用-I来附加指定其他的包含路径)去寻找这个头文件,如果找不到就会提示这个头文件不存在。
  • #include ""用来包含程序员写的头文件,编译器默认先在当前目录下寻找相应的头文件,如果没找到再到系统指定目录去寻找,如果还没找到则提示文件不存在。
使用原则
  • 系统自带的用<>
  • 程序员写的放在当前目录下用""
  • 程序员写的集中放专门存放头文件的目录下,在编译器中用-I参数寻找用<>

2.1.2 头文件包含在预处理时的处理方式

在预处理的时候,预处理器将所包含的头文件的内容原处展开替换这#include语句。
如下所示分别是同一目录下的.c文件和.h文件:

对其进行预编译生成.i文件后,.i文件的内容如下所示:

2.2 注释

我们在.c源文件中写的注释,预处理器在预处理阶段会将其擦除(在.c文件依然存在,在.i文件中不存在)。其实这也正顺应了注释是写给使用程序的人看的,而不是给编译器看的。如下所示:

2.3 条件编译

一般情况下,源程序中所有行都参与编译,但有时希望程序有多种配置,对一部分内容指定编译条件(如产品的调试版与正式版),这就是条件编译(conditional compile),条件编译有以下两种判定方法:

2.3.1 #if

这种判定方法类似于if...else...语句,格式为#if (条件表达式),它的判定标准是()中的表达式是true还是flase,在进行预编译的过程中,只会保留条件表达式为真的那部分内容,如下所示测试代码:

 1 #include <stdio.h>
2 #define DEBUG 1
3
4 int main(void)
5 {
6 #if(DEBUG == 1)
7 printf("Version: DEBUG!");
8 #elif(DEBUG == 2)
9 printf("Version: TEST!");
10 #else
11 printf("Version: LAST!");
12 #endif
13
14 return 0;
15 }

预编译结果如下所示:

我们也可以不在源码中对DEBUG进行宏定义,而在编译的时候可以用如下方法对其进行宏定义并指定值:

1 gcc -E -D DEBUG=2 test.c -o test.i

结果如下所示:

2.3.2 #ifdef

#ifdef XXX判定条件成立与否时主要是看XXX这个符号在本语句之前有没有被定义,只要定义了,判断就成立,并不关心XXX的宏值为多少。测试代码及结果参见下文3.1非带参宏的内容

三、宏定义

3.1 非带参宏

非带参宏主要结合条件编译使用,比较简单,其定义格式为

#define 宏名 替换列表(替换列表可有可无)

如下所示:

1 #define DEBUG
2 #define TEST 1
3 #define TEST2 TEST

宏定义的预处理

  1. 在预处理阶段由预处理器进行机械替换,而不做类型检查
  2. 宏定义替换会递归进行,直到替换出来的值本身不再是一个宏为止

如下是在STM32开发过程中常用的打印调试信息的一个调试代码段:

 1 #include <stdio.h>
2
3 #define USER_TIMER_DEBUG
4 #ifdef USER_TIMER_DEBUG
5 #define user_timer_printf(format, ...) printf( format "\r\n", ##__VA_ARGS__)
6 #define user_timer_info(format, ...) printf("[\ttimer]info:" format "\r\n", ##__VA_ARGS__)
7 #define user_timer_debug(format, ...) printf("[\ttimer]debug:" format "\r\n", ##__VA_ARGS__)
8 #define user_timer_error(format, ...) printf("[\ttimer]error:" format "\r\n",##__VA_ARGS__)
9 #else
10 #define user_timer_printf(format, ...)
11 #define user_timer_info(format, ...)
12 #define user_timer_debug(format, ...)
13 #define user_timer_error(format, ...)
14 #endif

预编译结果如图所示:

3.2 带参宏

宏可以带参数,称为带参宏。带参宏的使用和带参函数非常相似,只是在使用上和处理上有一些差异,其定义格式为:

#define 标识符(参数1,参数2,...,参数n) 替换列表

在定义带参宏时,每一个参数在宏体中引用时都必须加括号,最后整体再加括号,括号缺一不可。

不带括号的后果:

1 #include <stdio.h>
2 #define M(a, b) a * b
3 int main(void)
4 {
5 int result = M(2 + 3, 5)
6 printf("%d", result);
7 return 0;
8 }

如上测试代码,我们想得到(2 + 3) * 5的结果,但是由于宏在预处理的时候也是进行机械替换,int result = M(2 + 3, 5)变成了int result = 2 + 3 * 5,这及其容易出现逻辑上的错误

3.2.1 带参宏示例

1.MAX宏: 求2个数中较大的一个

#define MAX(a, b) (((a)>(b)) ? (a) : (b))

2.SEC_PER_YEAR宏 用宏定义表示一年中有多少秒

#define SEC_PER_YEAR (365*24*60*60UL)

这个宏需要注意的是

  1. 当一个数字直接出现在程序中时,它的是类型默认是int
  2. 一年有多少秒,这个数字超过了int类型存储的范围
  3. UL将其转为无符长整型

3.2.2 带参宏和带参函数的区别

1.时间与空间
  • 宏定义在预处理期间处理,进行简单的内容替换,无需额外空间
  • 函数是在编译期间处理的,调用时需要为形参分配空间并将实参的值赋给形参
2.执行速度
  • 宏只进行文本替换,函数运行阶段参数需要进行出入栈的操作,速度比宏慢
3.类型检查
  • 宏定义不会检查参数的类型,返回值也不会附带类型
  • 而数有明确的参数类型和返回值类型。当调用函数时编译器参数的静态类型检查,如果编译器发现实际传参和参数声明不同时会报警告或错误。
综合比较

宏和函数各有千秋,最大的特点是:用函数的时候程序员不太用操心类型不匹配因为编译器会检查,用宏的时候程序员必须注意实际传参和宏所希望的参数类型一致,或者自行加入类型检查,否则可能编译不报错但是运行有误。

如对MAX宏加入类型检查:

1 #define MAX(a, b) ({\
2 typeof(a) _a = (a); \
3 typeof(b) _b = (b); \
4 (void) (&_a == &_b);\
5 _a > _b ? _a : _b;})

测试代码:

 1 #include <stdio.h>
2
3 #define MAX(a, b) ({\
4 typeof(a) _a = (a); \
5 typeof(b) _b = (b); \
6 (void) (&_a == &_b);\
7 _a > _b ? _a : _b;})
8
9 #define MAX2(a, b) a > b ? a : b
10
11 int main(void)
12 {
13 int a = 2;
14 float b = 3.1;
15 int result2 = MAX2(a, b);
16 typeof(MAX(a, b)) result = MAX(a, b);
17 printf("result = %f\n", result);
18 printf("result2 = %d\n", result2);
19 return 0;
20 }

四、内联函数

内联函数本质上是函数,通过在函数定义前加inline关键字实现,是编译器负责处理的,可以做参数的静态类型检查。同时也有带参宏的展开特性,运行时没有调用开销。

4.1 与常规函数对比

  • 当函数体很短的时候,使用常规函数会造成很大的调用开销,内联函数采用原地展开的方式,没有调用开销
  • 当函数体长的时候,由于内联函数展开会降低寻址效率,所以长函数体不会使用内联函数
  • 内联函数本质上是函数,函数的性质内联函数都有

4.2 与宏的对比

  • 参数类型检查。编译过程中,编译器仍可以对其内联函数进行参数检查
  • 便于调试。函数支持的调试功内联函数同样支持,而宏不支持
  • 接口封装。有些内联函数可以用来封装一个接口,而宏不具备这个特性

4.3 noinlinealways_inline

当函数的函数体很小,而且被大量频繁调用,应该做内联展开时,就可以使用内联函数。但编译器会不会作内联展开,编译器也会有自己的权衡(不合理的内联函数会降低CPU寻址效率、函数运行效率、降低代码的可移植性…)。如果想告诉编译器一定要展开,或者不作展开,就可以使用noinlinealways_inline对函数作一个属性声明。

1 static inline int func(int *, int);//编译器权衡是否内联展开
2 static inline __attribute__((noinline)) int func(int *, int);//不内联展开
3 static inline __attribute__((always_inline)) int func(int *, int);//内联展开

static修饰呢是因为内联函数不一定会内联展开,当多个文件都包含同一个内联函数的定义时,如果没有static将函数的作用域限制在各自本地文件内,编译时就有可能报重定义错误

转自:https://blog.csdn.net/weixin_43955214/article/details/104255557

【C语言】预处理、宏定义、内联函数的更多相关文章

  1. C/C++之宏、内联函数和普通函数的区别

    内联函数的执行过程与带参数宏定义很相似,但参数的处理不同.带参数的宏定义并不对参数进行运算,而是直接替换:内联函数首先是函数,这就意味着函数的很多性质都适用于内联函数,即内联函数先把参数表达式进行运算 ...

  2. c++ _宏与内联函数

    第一部分:宏为什么要使用宏呢?因为函数的调用必须要将程序执行的顺序转移到函数所存放在内存中的某个地址,将函数的程序内容执行完后,再返回到转去执行该函数前的地方.这种转移操作要求在转去执行前要保存现场并 ...

  3. 特殊用途语言特性(默认实参/内联函数/constexpr函数/assert预处理宏/NDEBUG预处理变量)

    默认实参: 某些函数有这样一种形参,在函数的很多次调用中它们都被赋予一个相同的值,此时,我们把这个反复出现的值称为函数的默认实参.调用含有默认实参的函数时,可以包含该实参,也可以省略该实参. 需要特别 ...

  4. C++语言基础(7)-inline内联函数

    函数调用是有时间和空间开销的.程序在执行一个函数之前需要做一些准备工作,要将实参.局部变量.返回地址以及若干寄存器都压入栈中,然后才能执行函数体中的代码:函数体中的代码执行完毕后还要清理现场,将之前压 ...

  5. __inline定义的内联函数和宏的区别

    转自:http://blog.csdn.net/lw370481/article/details/7311668 函数与宏 #define TABLE_COMP(x) ((x)>0?(x):0) ...

  6. (转载)内联函数inline和宏定义

    (转载)http://blog.csdn.net/chdhust/article/details/8036233 内联函数inline和宏定义   内联函数的优越性: 一:inline定义的类的内联函 ...

  7. [C++] inline内联函数使用方法

    C++支持内联函数,目的是为了提高函数的执行效率,类似于C语言中的宏定义 内联函数在调用时将它在程序中的每个调用点展开,不用额外分配栈空间 内联函数的定义在一个源文件中出现一次,但在多个源文件中可以同 ...

  8. 【C语言天天练(二一)】内联函数

            引言:调用函数时,一般会由于建立调用.传递參数.跳转到函数代码并返回等花费掉一些时间,C语言的解决的方法是使用类函数宏.在C99中,还提出了第二种方法:内联函数.         内联 ...

  9. 嵌入式C语言自我修养 10:内联函数探究

    10.1 属性声明:noinline & always_inline 这一节,接着讲 __atttribute__ 属性声明,__atttribute__ 可以说是 GNU C 最大的特色.我 ...

  10. 从零开始学C++之从C到C++(二):引用、内联函数inline、四种类型转换运算符

    一.引用 (1).引用是给一个变量起别名 定义引用的一般格式:类型  &引用名 = 变量名: 例如:int a=1; int  &b=a;// b是a的别名,因此a和b是同一个单元 注 ...

随机推荐

  1. 深入理解ThreadLocal及其变种

    ThreadLocal 定义 ThreadLocal很容易让人望文生义,想当然地认为是一个"本地线程". 其实,ThreadLocal并不是一个Thread,而是Thread的局部 ...

  2. JavaBean , EJB , spring , POJO

    1996年 , 发布了java bean 1.00-A  当时的java bean有什么用呢 javaBean最初是为Java GUI的可视化编程实现的.你拖动IDE构建工具创建一个GUI 组件(如多 ...

  3. python+pytest接口自动化(4)-requests发送get请求

    python中用于请求http接口的有自带的urllib和第三方库requests,但 urllib 写法稍微有点繁琐,所以在进行接口自动化测试过程中,一般使用更为简洁且功能强大的 requests ...

  4. java中如何将嵌套循环性能提高500倍

    java中如何将嵌套循环性能提高500倍 转载请注明出处https://www.cnblogs.com/funnyzpc/p/15975882.html 前面 似乎上一次更新在遥远的九月份,按照既定的 ...

  5. LeetCode-268-丢失的数字

    丢失的数字 题目描述:给定一个包含 [0, n] 中 n 个数的数组 nums ,找出 [0, n] 这个范围内没有出现在数组中的那个数. 进阶: 你能否实现线性时间复杂度.仅使用额外常数空间的算法解 ...

  6. 记一次NAS故障分析(ZFS NFS)

    问题: 使用vdbench进行单层100w目录,每个目录30个文件,共3000w文件读写时,在创建文件得时候IO会出现断断续续得情况. 分析过程: 1.  nfs抓包分析 使用vdbench创建一个文 ...

  7. git flow 工作流程以及常用命令

    一.分支介绍 master 也是产品分支,只有一个,一般情况下不会在这个分支上进行代码操作 develop 只有一个,新特性的开发是基于 develop 开发的,但是不能直接在 develop 上进行 ...

  8. 七牛云cdn加速

    https://developer.qiniu.com/fusion/1228/fusion-quick-start https://blog.csdn.net/qq_27292113/article ...

  9. 支持向量机(SVM):用一条线分开红豆与绿豆

    算法原理 要找到一些线,这些线都可以分割红豆和绿豆,找到正确的方向或者斜率的那条线,确认马路的宽度,得到最优解--马路的中轴 超平面:在三维空间中,平面是两个点距离相同的点的轨迹.一个平面没有厚度,而 ...

  10. 分布式 PostgreSQL 集群(Citus)官方示例 - 实时仪表盘

    Citus 提供对大型数据集的实时查询.我们在 Citus 常见的一项工作负载涉及为事件数据的实时仪表板提供支持. 例如,您可以是帮助其他企业监控其 HTTP 流量的云服务提供商.每次您的一个客户端收 ...