learn_C_deep_14 (条件编译的基本使用与理解)

简介: learn_C_deep_14 (条件编译的基本使用与理解)

条件编译


1.条件编译如何使用?


       C语言的条件编译是一种在程序编译时根据条件选择不同代码段进行编译的技术。条件编译可以用于实现代码跨平台,开启或关闭某些功能,以及调试等情况。

#ifdef` 指令用于判断某个宏是否被定义,不论我们定义的宏为真为假,如果被定义则编译下面的代码块,否则跳过:

#include <stdio.h>
//#define PTINT
//#define PTINT 1  //真
//#define PTINT 0  //假
int main()
{
#ifdef PRINT //PRINT并没有被[定义],所以代码只保留#else部分
  printf("hello world!\n");
#else
  printf("Non Message!\n");//这句代码会被输出
#endif
  return 0;
}


与 `#ifdef` 相对应的是 `#ifndef`,它用于判断某个宏是否未被定义

#include <stdio.h>
//#define DEBUG
int main()
{
#ifndef DEBUG //DEBUG并没有被[定义],所以代码会输出#ifndef内容
  printf("hello debug\n");//这句代码会被输出
#else
  printf("hello release\n");
#endif
  return 0;
}


`#if` 指令可以根据表达式的值来判断是否编译下面的代码块。

// 单个条件的情况
#include <stdio.h>
//报错
//#define DEBUG
//定义了,为假
//#define DEBUG 0
//定义了,为真
#define DEBUG 1
int main()
{
#if DEBUG
  printf("hello bitworld\n");
#endif
    return 0;
}
//带else的情况
#include <stdio.h>
//报错
//#define DEBUG
//定义了,为假
#define DEBUG 0
//定义了,为真
//#define DEBUG 1
int main()
{
#if DEBUG
  printf("hello world\n");
#else
  printf("hello bit\n");
#endif
  return 0;
}
//多条件的情况
//代码一:
#include <stdio.h>
//#define DEBUG 0
//#define DEBUG 1
//#define DEBUG 2
#define DEBUG 3
int main()
{
#if DEBUG==0
  printf("hello world 0\n");
#elif DEBUG==1
  printf("hello world 1\n");
#elif DEBUG==2
  printf("hello world 2\n");
#else
  printf("hello else\n");
#endif
  return 0;
//代码二:
#include <stdio.h>
#define DEBUG -1
//#define DEBUG 0
//#define DEBUG 1
//#define DEBUG 2
int main()
{
#if DEBUG<0
  printf("hello world 0\n");
#elif DEBUG<1
  printf("hello world 1\n");
#elif DEBUG<2
  printf("hello world 2\n");
#else
  printf("hello else\n");
#endif
  return 0;
}


#if模拟#ifdef

#include <stdio.h>
#define DEBUG
int main()
{
#if defined(DEBUG)
  printf("hello debug\n");//这句话会被输出
#else
  printf("hello release\n");
#endif
  return 0;
}


#if模拟#ifndef

#include <stdio.h>
//#define DEBUG
int main()
{
#if !defined(DEBUG)
    printf("hello debug\n");//这句话会被输出
#else
    printf("hello release\n");
#endif
    return 0;
}


练习

#include <stdio.h>
#define C
#define CPP
int main()
{
#if defined(C) && defined(CPP)
  //#if (defined(C) && defined(CPP)) //建议使用这种方式
  printf("hello c&&cpp\n");//这句话会被输出
#else
  printf("hello other\n");
#endif
  return 0;
}


 这段代码使用了条件编译指令,定义了两个宏C和CPP,并且使用#if defined指令判断是否同时定义了这两个宏。


       当同时定义了C和CPP时,程序执行#if defined(C) && defined(CPP)条件满足,会输出"hello c&&cpp"。如果C和CPP没有同时定义,则条件不满足,会输出"hello other"。


在这段代码中,#if defined(C) && defined(CPP)和#if (defined(C) && defined(CPP))的效果是相同的。两者都是判断C和CPP是否都已经被定义了,如果都被定义,则条件满足,执行相应的代码块。


       注意,程序中使用#define指令定义的宏只是简单的文本替换,不会经过类型检查和语法检查。因此在使用宏时需要谨慎,避免出现不必要的错误。

#include <stdio.h>
#define C
//#define CPP
int main()
{
#if defined(C) || defined(CPP)
  //#if (defined(C) || defined(CPP))
  printf("hello c||cpp\n");//这句话会被输出
#else
  printf("hello other\n");
#endif
  return 0;
}


 这段代码与前面的稍有不同,它使用#if defined指令判断了C和CPP是否至少有一个被定义了,即#if defined(C) || defined(CPP)。当C和CPP中至少有一个被定义时,条件满足,会输出"hello c || cpp"。否则条件不满足,会输出"hello other"。


       和前面一样,注释掉#define CPP的语句,则只有C被定义,条件不满足,会输出"hello other"。


       需要注意的是,#if defined(C) || defined(CPP)和#if (defined(C) || defined(CPP))的效果也是相同的,这两种写法都是判断C和CPP是否至少有一个被定义了,如果有,则条件满足,执行相应的代码块。

#include <stdio.h>
#define C
#define CPP
int main()
{
#if !(defined(C) || defined(CPP))
  printf("hello c || cpp\n");//这句话会被输出
#else
  printf("hello other\n");
#endif
  return 0;
}


   这是一个C语言程序,定义了两个宏:C和CPP,并在主函数中使用了#if和#ifndef预编译指令。


       #if !(defined(C) || defined(CPP))表示如果C和CPP中的宏都未定义,则执行if语句内的代码,输出"hello c || cpp"。但是由于C和CPP都被定义了,所以if语句的条件不成立,所以会执行else语句内的代码,输出"hello other"。因此,这个程序的输出结果是"hello other"。

#include <stdio.h>
#define C
#define CPP
int main()
{
#if defined(C)
  #if defined (CPP)
    printf("hello CPP\n");//这句话会被输出
  #endif
  printf("hello C\n");//这句话会被输出
#else
  printf("hello other\n");
#endif
  return 0;
}


   这是一个C语言程序,其中使用了条件编译指令。程序定义了两个宏,分别为C和CPP。在main函数中,先判断宏C是否被定义,如果被定义,则进入C语言的条件编译指令中。在这个指令中,又判断宏CPP是否被定义,如果被定义,则输出"hello CPP\n"。无论宏CPP是否被定义,都会输出"hello C\n"。


       如果宏C没有被定义,则程序进入另一个条件编译指令中,输出"hello other\n"。因此,运行该程序,会输出"hello CPP\n"和"hello C\n"。

#include <stdio.h>
#define C
#define CPP
int main()
{
#if defined(C)
  printf("hello C\n");
#elif defined (CPP)
  printf("hello CPP\n");
#else
  printf("hello other\n");
#endif
  return 0;
}


    这是一个C语言程序,其中定义了两个宏C和CPP。在main函数中,同样使用了条件编译指令#if-elif-else-endif。在这个指令中,先判断宏C是否被定义,如果被定义,则输出"hello C\n"。如果宏C没有被定义,再判断宏CPP是否被定义,如果被定义,则输出"hello CPP\n"。如果宏C和CPP都没有被定义,则输出"hello other\n"。


       由于宏C和CPP都被定义了,因此运行该程序会输出"hello C\n"。


#if 和 #ifdef的区别


       #if和#ifdef是C预处理中的条件编译指令,它们的作用都是根据某个宏的定义情况来判断编译代码的方式。

  #if的语法形式为#if 常量表达式,它会首先对常量表达式进行求值,如果表达式的值为真,则编译这个#if与对应的#endif之间的代码,否则忽略这个#if与对应的#endif之间的代码。


#ifdef的语法形式为#ifdef 宏名,它用来判断某个宏是否定义了。如果宏被定义了,则编译#ifdef与对应的#endif之间的代码,否则忽略这个#ifdef与对应的#endif之间的代码。


       它们的主要区别在于#if是对表达式的求值结果进行判断,可以进行更加复杂的条件判断,而#ifdef只是判断宏是否被定义,不能进行表达式求值。另外,由于#if可以进行表达式求值,因此在某些情况下,#if比#ifdef会更加灵活和方便。


条件编译和宏替换谁先执行


宏替换先于条件编译执行。

 在C语言中,预处理器会先对源代码进行宏替换,将源代码中的宏名称替换为相应的宏定义,然后再对替换后的代码进行条件编译。也就是说,条件编译是在宏替换之后进行的。这也就意味着,在条件编译中对宏的定义或者值的改变,并不会影响到之前宏替换的结果。


在进行宏替换时,预处理器会根据宏定义中的替换列表来进行替换。如果宏定义中包含了其他宏的调用,那么预处理器会先对这些宏进行替换,然后再对当前宏进行替换。这种替换过程会一直持续下去,直到不包含其他宏的调用为止。


       总的来说,宏替换是在条件编译之前进行的,而且它是一个递归的过程,可以包含其他宏的调用,最终将源代码中的所有宏替换为相应的宏定义后,再对替换后的代码进行条件编译。


条件编译可以用来注释代码吗?推不推荐?


       条件编译可以用来注释代码,但不是最佳实践,因为它会影响到代码的可读性和可维护性。

在条件编译中,使用了类似于`#if 0 ... #endif`的语句来注释掉代码,它会导致这段代码被编译器忽略掉,从而达到注释的效果。如果这种语句对应的条件值为假,则这段代码将永远不会被编译,从而达到注释掉代码的效果。


       然而,使用条件编译来注释代码会使得代码难以阅读,特别是在复杂的代码环境中。因为代码注释完全可以通过标准注释语法来实现,这样会更加清晰易懂,也更易于维护。而使用条件编译来注释代码,会给读代码的人带来很大的困扰,因为他们不得不去寻找代码中存在的所有条件编译语句,以了解到底有哪些代码是被注释掉的。


       因此,推荐使用标准注释语法来进行注释,并尽可能避免使用条件编译来注释掉代码。这样会使代码更具可读性和可维护性。


结论:条件编译本质是让编译器进行代码裁剪。


2.为何要有条件编译?



3. 条件编译都在哪些地方用?



相关文章
西门子S7-1200编程实例,比较指令如何使用?
西门子S7-1200中的比较指令包括了值大小比较指令、是否在范围内指令以及有效性无效性检查指令。使用值大小比较指令,可以比较两个数据类型相同的数值的大小。
西门子S7-1200编程实例,比较指令如何使用?
|
监控 网络架构
CAN-TP传输协议详解
CAN-TP传输协议详解
CAN-TP传输协议详解
|
jenkins 测试技术 应用服务中间件
【专业测试技能】全流程掌握:部署测试环境的策略与实践
本文分享了关于部署测试环境的策略与实践。文章讨论了部署测试环境的全过程,包括服务如MySQL、Redis、Zookeeper等的部署,以及解决服务间的依赖和兼容问题。文中还介绍了使用Jenkins、Docker等工具进行部署的方法,并通过实战案例讲解了如何创建和管理Jenkins Job、配置代理服务器Nginx、进行前后端服务的访问和优化。最后,作者强调了提问的重要性,并鼓励大家通过互联网解决遇到的问题。
394 2
【专业测试技能】全流程掌握:部署测试环境的策略与实践
|
10月前
|
搜索推荐 API 开发者
Django框架和Flask框架的适用场景分别是什么?
总体而言,Django 更适合需要全面功能和大规模开发的场景,而 Flask 则更适合灵活性要求高、小型项目或特定需求的开发。当然,具体的选择还应根据项目的具体情况、团队的技术能力和偏好等因素来综合考虑。在实际应用中,开发者可以根据项目的特点和需求,灵活选择使用这两个框架,或者结合它们的优势来构建更强大的 Web 应用程序。
331 64
|
8月前
|
数据挖掘 测试技术 项目管理
2025年测试用例管理看这一篇就够了 ----Codes 开源免费、全面的测试管理解决方案
Codes 是国内首款重新定义 SaaS 模式的开源项目管理平台,支持云端认证、本地部署、全部功能开放,并且对 30 人以下团队免费。它通过整合迭代、看板、度量和自动化等功能,简化测试协同工作,使敏捷测试更易于实施。并提供低成本的敏捷测试解决方案,如同步在线离线测试用例、流程化管理缺陷、低代码接口自动化测试和 CI/CD,以及基于迭代的测试管理和测试用时的成本计算等,践行敏捷测试。
2025年测试用例管理看这一篇就够了 ----Codes 开源免费、全面的测试管理解决方案
|
11月前
|
数据可视化 IDE Linux
【Python篇】PyQt5 超详细教程——由入门到精通(序篇)
【Python篇】PyQt5 超详细教程——由入门到精通(序篇)
1584 3
|
Unix Linux Shell
Sphinx是一个Python文档生成工具,它可以解析reStructuredText或Markdown格式的源代码注释,并生成多种输出格式,如HTML、LaTeX、PDF、ePub等。
Sphinx是一个Python文档生成工具,它可以解析reStructuredText或Markdown格式的源代码注释,并生成多种输出格式,如HTML、LaTeX、PDF、ePub等。
|
算法
【改进灰狼优化算法】改进收敛因子和比例权重的灰狼优化算法【期刊论文完美复现】(Matlab代码实现)
【改进灰狼优化算法】改进收敛因子和比例权重的灰狼优化算法【期刊论文完美复现】(Matlab代码实现)
642 1
|
JSON 算法 JavaScript
JSON比较:如何轻松判断数据格式是否相同?
JSON比较:如何轻松判断数据格式是否相同?
780 0
|
安全 网络安全 数据安全/隐私保护