Summary

1)条件编译的行为相似于C语言中的if else;条件编译是预编译批示命令,用于管制是否编译某段代码

2)预编译器依据条件编译指令有抉择的删除代码,所以编译器不晓得代码分支的存在

3)if else在运行期进行分支判断,肯定会被编进指标代码条件编译指令预编译期进行分支判断,可能产生不同的代码段,因而编进指标代码的代码段不确定

4)能够通过命令行定义宏gcc -Dmacro=val file.cgcc -Dmacro file.c

5)#include 实质 是将曾经存在的文件内容插入到以后文件中;#include的间接蕴含同样会产生嵌入文件内容操作。条件编译指令#ifndef _FILE_H_ #define _FILE_H #endif能够解决这种头文件反复蕴含的谬误。

6)条件编译指令只是能够在同一个.c文件中,避免反复蕴含;如果头文件中有了符号的定义,然而在一个工程的不同c文件里都进行了include,这时候编译这两个文件也会有反复定义的谬误(因为两份include在各自的c文件里都定义了一个global,在同一个全局作用域里定义了同名的symbol)。所以,头文件中只申明、不定义

7)工程中的条件编译次要用于:不同的产品线共用一份代码辨别编译产品的调试版和公布版

条件编译剖析

条件编译的行为相似于C语言中的if...else...
条件编译是预编译批示命令,用于管制是否编译某段代码;

1、条件编译于if...else的区别

  • 预编译器依据条件编译指令有抉择的删除代码,所以编译器不晓得代码分支的存在
  • if...else语句在运行期进行分支判断;条件编译指令预编译期进行分支判断
  • 能够通过命令行定义宏gcc -Dmacro=val file.c或者gcc -Dmacro file.c
#define C1int main(){    #if (C==1)        printf("if true\n");    #else        printf("if false\n");    #endif        return 0;}
gcc -E test.c -o test.i// 单步编译后失去的两头文件# 1 "test.c"# 1 "<built-in>"# 1 "<command-line>"# 1 "test.c"int main(){        printf("if true\n");    return 0;}

剖析:

  • 通过预编译期的解决后,失去的两头文件中#if #else #endif都被删掉了,这也阐明了下面的预编译器依据条件编译指令有抉择的删除代码,所以编译器在拿到两头.i文件后,基本就不晓得这些代码分支的存在。
  • 也阐明,#if #else #endif是在预编译期进行判断的,而if else是在运行期才会判断
  • // 上述代码中去掉#define C 1// 应用命令来定义宏gcc -DC=1 test.c编译后的运行后果为:if truegcc -DC test.cgcc -DC test.c编译后的运行后果为:if true

应用#ifdef #else #endif进行预编译分支判断:

  • #ifdef C  printf("yes, defined");#else  printf("no, undefined");#endif以上代码进行编译:gcc -DC test.c输入:yes, definedgcc test.c输入:no, undefined

2、条件编译解决头文件反复蕴含的编译谬误

  • #include 实质 是将曾经存在的文件内容插入到以后文件中
  • #include的间接蕴含同样会产生嵌入文件内容操作。
对以上代码进行单步编译gcc -E test.c -o test.igcc -S test.i -o test.s

剖析:单步编译后,在两头.i文件中,global的定义呈现了2次,后续的编译过程天然会呈现重定义的谬误。

反复蕴含的解决形式:在头文件test.h和global.h中加上条件编译指令

#ifndef _HEADER_FILE_H_#define _HEADER_FILE_H_// SRC#endif

留神条件编译指令只是能够在同一个.c文件中,避免反复蕴含;如果头文件中有了符号的定义,然而在一个工程的不同c文件里都进行了include,这时候编译这两个文件也会有反复定义的谬误(因为两份include在各自的c文件里都定义了一个global,在同一个全局作用域里定义了同名的symbol)。所以,头文件中只申明、不定义
示例代码:

仍旧是下面的test.c test.h global.h,再加一个test2.c文件,其中#include "test.h"执行:gcc test.c test2.c输入:multiple definition of 'global'

3、条件编译工程中的利用

  • if else编译器解决,必然被编译进指标代码#if #else #endif预编译器解决,能够按不同的条件编译不同的代码段,因此会产生不同的指标代码
  • 工程中的条件编译次要用于:不同的产品线共用一份代码辨别编译产品的调试版和公布版

本文总结自“狄泰软件学院”唐佐林老师《C语言进阶课程》。
如有错漏之处,恳请斧正。