22_条件编译使用分析

关键词:条件编译的本质、#include的本质、 条件编译的意义

0 基本概念:

1) 条件编译的行为类似于C语言中的if...else...;
2)条件编译是预编译指示命令,用于控制是否编译某段代码;
3)可以利用预处理器调整代码、删除代码的操作。

//#include<stdio.h>
#define C 1

int main()
{
    const char* s;
    #if(1 == C)
        s = "this is first printf...\n";
    #else
        s = "this is second printf...\n";
    #endif

//  printf("%s:", s);

    return 0;
}

编译器处理后的代码为:

# 1 "1.c"
# 1 "<built-in>"
# 1 "<command-line>"
# 1 "/usr/include/stdc-predef.h" 1 3 4
# 1 "<command-line>" 2
# 1 "1.c"

int main()
{
 const char* s;

  s = "this is first printf...\n";

 return 0;
}

总结:#if...#else...:用来告诉预处理器保留哪一部分代码,删除哪一部分代码。

1 条件编译的本质

1) 预处理器根据条件编译指令有选择的删除代码
2) 编译器不知道代码分支的存在
3) if...else...语句在运行期进行分支判断
4) 条件编译指令在预编译期进行分支判断
5) 可以通过命令定义宏,如下代码:

gcc -Dmacro=value file.c
or
gcc -Dmacro file.c
  • 方式一:
#include <stdio.h>

int main()
{
    const char* s;
    
    #if( C == 1 )
        s = "this is first printf...\n";
    #else
        s = "this is second printf...\n";
    #endif
    
    printf("%s", s);

    return 0;
}

单步编译:gcc -DC=1 -E 1.c -o 1.i
输出结果:

# 2 "1.c" 2

int main()
{
 const char* s;


  s = "this is first printf...\n";




 printf("%s", s);

 return 0;
}
  • 方式二:
#include <stdio.h>

int main()
{
    const char* s;
    
    #ifdef C
        s = "this is first printf...\n";
    #else
        s = "this is second printf...\n";
    #endif
    
    printf("%s", s);

    return 0;
}

单步编译:gcc -DC -E 1.c -o 1.i
输出结果:

int main()
{
 const char* s;


  s = "this is first printf...\n";




 printf("%s", s);

 return 0;
}

2 #include的本质

  • #include的本质是将已经存在的文件内容嵌入到当前文件中
  • #include间接包含同样会产生嵌入文件内容的操作
  • 条件编译可以解决头文件重复包含的编译错误
#include _HEADER_FILE_H_
#define _HEANDER_FILE_H_

// source code 

#endif

3 条件编译的意义

  • 条件编译使得我们可以按照不同的条件编译不同的代码段,因而可以产生不同的目标代码
  • #if...#else...#endif预编译器处理if...else...语句被编译器处理,必然被编译进目标代码
  • 实际工程中条件编译主要用于以下两种情况:
    不同的产品线公用一份代码
    区分编译产品的调试版和发布版
#include <stdio.h>
#include "product.h"

// 通过条件编译定义当前版本是否为dubug版本
#if DEBUG
  #define LOG(s) printf("[%s:%d] %s\n", __FILE__, __LINE__, s)
#else
  #define LOG(s) NULL
#endif

// 通过条件编译定义当前版本是否为高端产品
#if HIGH
void f()
{
    printf("This is the high level product!\n");
}
#else
void f()
{
}
#endif

int main()
{
    LOG("Enter main() ...");
    
    f();

    printf("1. Query Information.\n");
    printf("2. Record Information.\n");
    printf("3. Delete Information.\n");   

    #if HIGH
      printf("4. High Level Query.\n");
      printf("5. Mannul Service.\n");
      printf("6. Exit.\n");
    #else
      printf("4. Exit.\n");
    #endif
   
    LOG("Exit main() ...");

    return 0;
}

product.h

#define DEBUG 1
#define HIGH  1

输入测试命令:gcc 2.c
输出结果:

[2.c:25] Enter main() ...
This is the high level product!
1. Query Information.
2. Record Information.
3. Delete Information.
4. High Level Query.
5. Mannul Service.
6. Exit.
[2.c:41] Exit main() ...

4 小结

  • 通过编译器命令行能够定义预处理器使用的宏
  • 条件编译可以避免重复包含同一个头文件
  • 条件编译在工程开发中可以区别不同产品线的代码
  • 条件编译可以定义产品的发布版和调试版

声明:此文章为本人在学习狄泰软件学院《C语言深度解析》所做的笔记,文章中包含狄泰软件资料内容一切版权归狄泰软件所有!

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • C中的预编译宏定义 2009-02-10 作者: infobillows 来源:网络 在将一个C源程序转换为可执行...
    白水灬煮一切阅读 1,669评论 0 5
  • 目录 一.预处理的工作方式... 3 1.1.预处理的功能... 3 1.2预处理的工作方式... 3 二.预处理...
    朱森阅读 1,443评论 0 2
  • 我打碎了一只最远的瓷碗揽回来更远的生命荷叶捧着稻米鼻子闻我打碎了一只精美的瓷碗拦回去更精美的生命荷叶捧着稻米鼻子闻...
    Amaorent阿毛的空瓶子阅读 371评论 15 12
  • 还有364天就到2017年了,时间总是过得如此匆匆,2016年的元旦节仿佛就像发生在昨天一样。
    0fd493c98e41阅读 227评论 0 0
  • 文人无双(原创)/微信公众号“心际花园” 周六和醒悟一起,看了《煎饼侠》,专门买的VIP坐席。 电影讲的是一个拍电...
    心际花园阅读 293评论 0 1