【转】#ifndef作用

简介: 定义  #ifndef x   #define x   ...   #endif   这是宏定义的一种,它可以根据是否已经定义了一个变量来进行分支选择,一般用于调试等等.实际上确切的说这应该是预处理功能中三种(宏定义,文件包含和条件编译)中的一种----条件编译。
定义   #ifndef x
   #define x
  ...
  #endif
  这是 定义的一种,它可以根据是否已经定义了一个变量来进行分支选择,一般用于调试等等.实际上确切的说这应该是预处理功能中三种(宏定义,文件包含和条件编译)中的一种----条件编译。
C语言在对程序进行编译时,会先根据 预处理命令进行“预处理”。C语言 编译系统包括预处理,编译和链接等部分。
   #ifndef x //先测试x是否被宏定义过
   #define x
  程序段1 //如果x没有被宏定义过,定义x,并编译程序段1
   #endif
  程序段2 //如果x已经定义过了则编译程序段2的语句,“忽视”程序段1。
  条件指示符#ifndef 的最主要目的是防止 头文件的重复包含和编译。了解:条件编译当然也可以用条件语句来实现。
但是用条件语句将会对整个源程序进行编译,生成的目标代码程序很长,而采用条件编译,则根据条件只编译其中的程序段1或程序段2,生成的目标程序较短。如果条件选择的程序段很长,采用条件编译的方法是十分必要的。

编辑本段补充一些内容

  #ifndef 标示1
//判断"标示1"是否定义,如果被定义则返回假,如果没有被定义则返回真。
  /**********************************/
  语句1 #ifndef 标示1
  语句2 #define 标示1
  语句3 #endif
  语句4 ……
  语句5 ……
  该段代码意思是:如果标示1没有被定义,则重定义标示1,即执行语句2、语句3、语句4、……;如果标示1已经被定义,则直接跳过语句2、语句3,直接执行语句4、语句5、……

  /***********************************/
  备注:#ifndef 和 #endif 要一起使用,如果丢失#endif,可能会报错。
  千万不要忽略了头文件中的#ifndef,这是一个很关键的东西。比如你有两个C文件,这两个C文件都include了同一个头文件。而编译时,这两个C文件要一同编译成一个可运行文件,于是问题来了,大量的声明冲突。

  还是把头文件的内容都放在#ifndef和#endif中吧。不管你的头文件会不会被多个文件引用,你都要加上这个。一般格式是这样的:

  #ifndef <标识>
  #define <标识>
  ......
  ......
  #endif
  <标识>在理论上来说可以是自由命名的,但每个头文件的这个“标识”都应该是唯一的。标识的命名规则一般是头文件名全大写,前后加下划线,并把文件名中的“.”也变成下划线,如: stdio.h
  #ifndef _STDIO_H_
  #define _STDIO_H_
  ......
  #endif

编辑本段详细讨论#ifndef的作用

  在c语言中,对同一个变量或者函数进行多次声明是不会报错的。所以如果h文件里只是进行了声明工作,即使不使用#
ifndef宏定义,多个c文件包含同一个h文件也不会报错。
  但是在c++语言中,#ifdef的作用域只是在单个文件中。所以如果h文件里定义了全局变量,即使采用#ifdef宏定义,多个c文件包含同一个h文件还是会出现全局变量重定义的错误。

  使用#ifndef可以避免下面这种错误:如果在h文件中定义了全局变量,一个c文件包含同一个h文件多次,如果不加#ifndef宏定义,会出现变量重复定义的错误;如果加了#ifndef,则不会出现这种错误。

编辑本段条件编译的命令的三种形式

  “
条件编译”命令允许对程序中的内容选择性地编译。即可以根据一定的条件选择是否编译。
  条件编译的命令主要有以下几种
   形式1
  # i f d e f 标识符
  程序段 I
  # el s e
  程序段 2
  # en di f
  它的作用是当 “ 标识符”已经由# d e f i n e定义过了。则编译“ 程序段 1 ” 。 否则编译“
程序段 2 ” 。其中如果不需要编译“ 程序段 2 ”。则上述形式可以变换 为:
  # i f d e f标识符
  程序段 1
  # en di f
   形式2
  # i f n d e f 标识符
  # define 标识符
  程序段 1
  # el s e
  程序段 2
  #e n di f
  它的作用是当 “ 标识符 没有由# d e f i n e定义过。 则编译“程序段 1”。否则编译“程序段
2” 。同样当无“ 程序段2 ”时。则上述形式变换为:
  # i f n d e f标识符
  # define 标识符
  程序段 1
  #e n di f
   形式3
  # if表达式
  程序段 1
  #el s e
  程序段 2
  #endif
  它的作用是 当“表达式”值为真时。编译程序段1。否则则编译程序段
  2。同样 当无程序段 2时,则上述形式变换 为 :
  # 表达式
  程序段 1
  #e n di f
  以上三种形式的条件编译预处理
  结构都可以嵌套使用。 当# e l s e后嵌套 # i f 时,可以使用预处理命令 # e l i f ,
它相
  当于 # e l s e# i f 。在程序中使用条件编译主要是为了方便程序的调试和移植。
相关文章
|
安全 开发工具 git
【Git】—— 分支管理策略
【Git】—— 分支管理策略
261 0
|
人工智能 Serverless 数据处理
极速启动,函数计算弹性降本能力再升级
在数字化转型的大潮中,云计算成为推动创新和优化业务流程的关键力量。作为阿里巴巴集团的核心产品之一,函数计算(Function Compute)引领着 Serverless 计算的新时代。本文将深入探讨函数计算如何通过技术革新实现提效降本,以及其在 AI 业务、数据处理和 Web 应用等多个领域的广泛应用。
556 31
|
数据可视化 数据库
通义灵码个人版新功能最佳实践测评
作为一名软件开发工程师,我使用通义灵码个人版的@workspace和@terminal功能,快速熟悉新的电商项目代码并实现新功能,效率提升了约30%。通过自动识别项目结构、代码分析、智能注释和自动补全等功能,大幅减少了查找资料和调试的时间,使开发流程更加顺畅,专注于业务逻辑实现。
331 1
|
网络协议 Linux Apache
如何查看Linux服务器中所有正在运行的进程服务?
有许多方法和工具可以查看 Linux 中所有正在运行的服务。大多数管理员会在 System V(SysV)初始化系统中使用
9321 1
|
Ubuntu Linux
内核实验(八):实现O-NONBLOCK非阻塞读写
本文通过修改之前的内核模块代码,介绍了如何在Linux内核中实现O_NONBLOCK非阻塞读写机制,并通过在Qemu虚拟机上的测试验证了非阻塞读写操作的正确性。
231 0
内核实验(八):实现O-NONBLOCK非阻塞读写
|
存储 JSON OLAP
Hologres支持哪些数据格式?
【8月更文挑战第20天】Hologres支持哪些数据格式?
399 1
在Linux中,如何进行备份或归档文件(tar 命令)?
在Linux中,如何进行备份或归档文件(tar 命令)?
|
存储 移动开发 测试技术
苹果个人付费开发者证书申请及使用图文详解
上篇文章《iOS真机安装WebDriverAgent图文详解》中提到如果使用付费版开发者证书去编译安装WDA,那么就不需要在手机上点击信任,这样更方便自动化测试的执行,今天就给大家详细分享一下如何申请个人付费开发者证书以及如何使用。
3018 0
苹果个人付费开发者证书申请及使用图文详解
|
Kubernetes IDE Serverless
Serverless 应用引擎操作报错合集之在阿里函数计算中,SD Controlnet Depth 运行过程中出现错误“urllib3 v2.0 only supports OpenSSL 1.1.1+”如何解决
Serverless 应用引擎(SAE)是阿里云提供的Serverless PaaS平台,支持Spring Cloud、Dubbo、HSF等主流微服务框架,简化应用的部署、运维和弹性伸缩。在使用SAE过程中,可能会遇到各种操作报错。以下是一些常见的报错情况及其可能的原因和解决方法。
659 3
|
JavaScript 前端开发 索引
学习了解JS的classlist
学习了解JS的classlist
308 0