假如现在要开发一个C语言程序,让它输出红色的文字,并且要求跨平台,在 Windows 和 Linux 下都能运行,怎么办呢?

这个程序的难点在于,不同平台下控制文字颜色的代码不一样,我们必须要能够识别出不同的平台。

Windows 有专有的宏_WIN32,Linux 有专有的宏__linux__,以现有的知识,我们很容易就想到了 if else,请看下面的代码:


  1. #include <stdio.h>
  2. int main(){
  3. if(_WIN32){
  4. system("color 0c");
  5. printf("http://c.biancheng.net\n");
  6. }else if(__linux__){
  7. printf("\033[22;31mhttp://c.biancheng.net\n\033[22;30m");
  8. }else{
  9. printf("http://c.biancheng.net\n");
  10. }
  11. return 0;
  12. }

但这段代码是错误的,在 Windows 下提示 __linux__ 是未定义的标识符,在 Linux 下提示 _Win32 是未定义的标识符。对上面的代码进行改进:


  1. #include <stdio.h>
  2. int main(){
  3. #if _WIN32
  4. system("color 0c");
  5. printf("http://c.biancheng.net\n");
  6. #elif __linux__
  7. printf("\033[22;31mhttp://c.biancheng.net\n\033[22;30m");
  8. #else
  9. printf("http://c.biancheng.net\n");
  10. #endif
  11. return 0;
  12. }

#if、#elif、#else 和 #endif 都是预处理命令,整段代码的意思是:如果宏 _WIN32 的值为真,就保留第 4、5 行代码,删除第 7、9 行代码;如果宏 __linux__ 的值为真,就保留第 7 行代码;如果所有的宏都为假,就保留第 9 行代码。

这些操作都是在预处理阶段完成的,多余的代码以及所有的宏都不会参与编译,不仅保证了代码的正确性,还减小了编译后文件的体积。

这种能够根据不同情况编译不同代码、产生不同目标文件的机制,称为条件编译。条件编译是预处理程序的功能,不是编译器的功能。

条件编译需要多个预处理命令的支持,下面一一讲解。

#if 的用法

#if 用法的一般格式为:

#if 整型常量表达式1
    程序段1
#elif 整型常量表达式2
    程序段2
#elif 整型常量表达式3
    程序段3
#else
    程序段4
#endif

它的意思是:如常“表达式1”的值为真(非0),就对“程序段1”进行编译,否则就计算“表达式2”,结果为真的话就对“程序段2”进行编译,为假的话就继续往下匹配,直到遇到值为真的表达式,或者遇到 #else。这一点和 if else 非常类似。

需要注意的是,#if 命令要求判断条件为“整型常量表达式”,也就是说,表达式中不能包含变量,而且结果必须是整数;而 if 后面的表达式没有限制,只要符合语法就行。这是 #if 和 if 的一个重要区别。

#elif 和 #else 也可以省略,如下所示:


  1. #include <stdio.h>
  2. int main(){
  3. #if _WIN32
  4. printf("This is Windows!\n");
  5. #else
  6. printf("Unknown platform!\n");
  7. #endif
  8. #if __linux__
  9. printf("This is Linux!\n");
  10. #endif
  11. return 0;
  12. }

#ifdef 的用法

#ifdef 用法的一般格式为:

#ifdef  宏名
    程序段1
#else
    程序段2
#endif

它的意思是,如果当前的宏已被定义过,则对“程序段1”进行编译,否则对“程序段2”进行编译。

也可以省略 #else:

#ifdef  宏名
    程序段
#endif

VS/VC 有两种编译模式,Debug 和 Release。在学习过程中,我们通常使用 Debug 模式,这样便于程序的调试;而最终发布的程序,要使用 Release 模式,这样编译器会进行很多优化,提高程序运行效率,删除冗余信息。

为了能够清楚地看到当前程序的编译模式,我们不妨在程序中增加提示,请看下面的代码:


  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. int main(){
  4. #ifdef _DEBUG
  5. printf("正在使用 Debug 模式编译程序...\n");
  6. #else
  7. printf("正在使用 Release 模式编译程序...\n");
  8. #endif
  9. system("pause");
  10. return 0;
  11. }

当以 Debug 模式编译程序时,宏 _DEBUG 会被定义,预处器会保留第 5 行代码,删除第 7 行代码。反之会删除第 5 行,保留第 7 行。

#ifndef 的用法

#ifndef 用法的一般格式为:

#ifndef 宏名
    程序段1 
#else 
    程序段2 
#endif

与 #ifdef 相比,仅仅是将 #ifdef 改为了 #ifndef。它的意思是,如果当前的宏未被定义,则对“程序段1”进行编译,否则对“程序段2”进行编译,这与 #ifdef 的功能正好相反。

三者之间的区别

最后需要注意的是,#if 后面跟的是“整型常量表达式”,而 #ifdef 和 #ifndef 后面跟的只能是一个宏名,不能是其他的。

例如,下面的形式只能用于 #if:


  1. #include <stdio.h>
  2. #define NUM 10
  3. int main(){
  4. #if NUM == 10 || NUM == 20
  5. printf("NUM: %d\n", NUM);
  6. #else
  7. printf("NUM Error\n");
  8. #endif
  9. return 0;
  10. }

运行结果:
NUM: 10

再如,两个宏都存在时编译代码A,否则编译代码B:


  1. #include <stdio.h>
  2. #define NUM1 10
  3. #define NUM2 20
  4. int main(){
  5. #if (defined NUM1 && defined NUM2)
  6. //代码A
  7. printf("NUM1: %d, NUM2: %d\n", NUM1, NUM2);
  8. #else
  9. //代码B
  10. printf("Error\n");
  11. #endif
  12. return 0;
  13. }

运行结果:
NUM1: 10, NUM2: 20

#ifdef 可以认为是 #if defined 的缩写。

C语言#if、##ifdef、#ifndef的用法详解,C语言条件编译详解相关推荐

  1. 怎么在c语言中使用常用对数,【用法】在C语言中使用对数函数的方法

    这篇文章主要介绍了在C语言中使用对数函数的方法,包括以e为底和以10为底的对数计算,需要的朋友可以参考下. C语言log()函数:返回以e为底的对数值 头文件: #include log() 函数返回 ...

  2. 【Scripts系列】之Makefile中条件分支ifeq/else/endif/else ifeq/ifneq/ifdef/ifndef用法详解

    DATE: 2019-2-22 前言       条件分支的用法在任何编程语言和脚本以及逻辑中都广泛使用.前文讲述过C语言中预处理的条件编译#ifdef/#if/#elif/#else/#endif用 ...

  3. C语言中宏定义宏编译的使用#if,#ifdef,#ifndef,#elif,#else,#endif,defined使用详解,可以实现和if else一样的效果

    #define            定义一个预处理宏 #undef            取消宏的定义 #if                   编译预处理中的条件命令,相当于C语法中的if语句 ...

  4. c++ 预处理命令 #if, #ifdef, #ifndef, #else, #elif, #endif用法

    #if, #ifdef, #ifndef, #else, #elif, #endif用法 #if, #ifdef, #ifndef, #else, #elif, #endif #if, #ifdef, ...

  5. #if, #ifdef, #ifndef, #else, #elif, #endif的用法

    #ifdef的用法 灵活使用#ifdef指示符,我们可以区隔一些与特定头文件.程序库和其他文件版本有关的代码. 代码举例:新建define.cpp文件 #include "iostream. ...

  6. c语言fputc输入字符串,C语言fgetc和fputc函数用法详解(以字符形式读写文件)

    C语言fgetc和fputc函数用法详解(以字符形式读写文件),文件,字符,指针,函数,字节 C语言fgetc和fputc函数用法详解(以字符形式读写文件) 易采站长站,站长之家为您整理了C语言fge ...

  7. 计算机科学类专升本复习之“C语言fgetc和fputc函数用法”详解(初稿)

    C语言fgetc和fputc函数用法详解(以字符形式读写文件) 在C语言中,"读写"文件比较灵活,既可以每次"读写一个字符",也可以"读写一个字符串& ...

  8. c语言malloc用法 数组,详解C语言用malloc函数申请二维动态数组的实例

    详解C语言用malloc函数申请二维动态数组的实例 C语言在程序运行中动态的申请及释放内存十分方便,一维数组的申请及释放比较简单. Sample one #include int main() { c ...

  9. 武林c语言,详解C语言中条件编译

    预处理器提供条件编译,程序的不同部分可以在不同的条件下编译,从而产生不同的目标代码文件,这对于程序移植和调试非常有用,本文是武林技术频道小编给为大家带来的详解中条件编译,一起来了解一下吧! 通常情况, ...

最新文章

  1. 数组指针和指针数组,函数指针和指针函数,常量指针和指针常量,常量引用
  2. 识别Gradle约定
  3. 操作系统学习笔记目录(暂时不全223)
  4. 计算机护理职称考试报名时间2015,护理职称考试怎么报名?
  5. html文件用safari打开方式,在html中打开(在Safari中)
  6. 如何简单解释 MapReduce算法
  7. 工程管理 -- makefile
  8. IDEA代码格式化快捷键失效的解决办法
  9. 2022年玻璃包装容器行业市场发展环境分析预测及下游需求规模增长率研究预测
  10. (C语言)有n人围成一圈,顺序排号。从第1个人开始报数(从1到3报数),凡报到3的人退出圈子,问最后留下的是原来的第几号的那位。
  11. 2020年电商行业的前景
  12. ch01变量和数据结构
  13. 【OPENCV_系列电子PDF图书连载】计算机视觉从入门到精通完整学习路线专栏
  14. [RK3288][Android6.0] 调试笔记 --- Activity多次调用finish()引起的Warning
  15. 华为MA5626配置为普通交换机的方法
  16. Graphviz安装向导及入门指南
  17. python添加两段音乐_Python实现音乐的剪辑
  18. L101. 对称二叉树/镜像二叉树
  19. ThreeJs做智慧城市项目后记
  20. C++,中文乱码问题解决办法

热门文章

  1. 计算机科学与技术专业软考考什么,计算机专业软考考什么
  2. FTP登录不上 显示“找不到元素” windows无法访问此文件夹,请确保输入的文件名是否正确,并且您有权访问此文件
  3. Replication进阶(三) 复制心跳详解
  4. 安卓版谷歌浏览器怎么样 Android版Chrome评测
  5. [激光原理与应用-60]:激光器 - 光学 - 光的四大理论框架与其层次:几何光学、波动光学、电磁光学、电子光学
  6. 固态硬盘安装--系统迁移--设置引导启动项
  7. 【文献阅读】受山体阴影影响的冰湖制图方法研究(Li JunLi等人,2018.09,IJRS)
  8. 好用的PERL正则表达式在线测试工具
  9. 【C/C++】scanf,printf 函数
  10. 【本人秃顶程序员】Redis 这么火,它都解决了哪些问题?