当前位置: 首页 > news >正文

#ifdefine #define #endif (避免头文件被重复包含的真正含义)

宏定义

首先在谈论正式话题之前,需要先介绍一个基础概念,也是前提,那就是宏定义。

#define demo 1
#define PI 3.14

我们都知道这样会将demo 在预处理阶段替换或者说展开为1,Pi 替换为3.14。
#define 宏定义一个标识符来表示一个常量。预处理所执行的操作就是简单的“文本”替换。

#include <stdio.h>也是这样的,即在预处理的时候先单纯地用头文件stdio.h中所有的“文本”内容替换程序中#include <stdio.h>这一行,然后再进行正式编译。

参考这个链接:

http://c.biancheng.net/view/187.html

条件编译

好了,知道这个概念以后我们就可以引入接下来这个概念,条件编译。

#if defined() || defined() || ...
#ifdef
#ifndef
#else if
#else
#endif

注意,我只是把可能遇见的所有条件编译宏列了一下,并不代表真正写的顺序就是这样。

至于#if defined#ifdef 的区别:
#if defined 就是#if 的一种用法,只不过后面跟了defined
对于#if后面需要是一个表达式,如果表达式为1则调用#if下面的代码。
比如:

#if defined(AA) || defined(BB) //和常规if一样,只是判断当前是否定义了AA,BB,用的是英文的过去时。defined 已经定义了。
#define  WIDE_IN_SENSOR_ZOOM_LUX    390  //你的代码,可以写很多
code...
#endif

对于#ifdef后面需要的只是这个值有没有用#define定义,并不关心define的这个值是0还是1。
比如在上面这里这个例子,就可以接着这样写:

#if defined(AA) || defined(BB) //和常规if一样,只是判断当前是否定义了AA,BB,用的是英文的过去时。defined 已经定义了。
#define  WIDE_IN_SENSOR_ZOOM_LUX    390  //你的代码,可以写很多
code...
#ifdef WIDE_IN_SENSOR_ZOOM_LUX //上面已经定义了WIDE_IN_SENSOR_ZOOM_LUX ,接着执行下面代码
code..
#endif //时刻注意每个条件都应该一个endif
#endif

还有一个区别就是,#ifdef 只能判断一个宏,如果判断多个宏只能用 #if defined() || defined () || defined ()

避免头文件被多次包含(预处理时重复include)

那么最后回到我们的正题,一般我们能在头文件看到最开始这样定义

#ifndef A_H //这个地方不需要和头文件名一样!理解这句话你就明白避免重复包含的真正含义了,而不是单纯记一个八股文知识点。
#defined A_H
code...
#endif

上面是通常的写法,一般重复包含指的是,
我们在a.cpp中

#include "a.h" 
#include "b.h"

但是在a.h中我们又

#include "b.h"

这样在预处理阶段,展开#include "a.h" 时候会递归展开b.h, 然后#include "b.h"又展开一遍,这就是重复包含了。
虽然我们自己在写程序的时候,一般不会编译报错。因为我们写的都是小程序,并不是大型的工程,但是在大型工程中,可能会因为重复包含而编译出错。


回到上面,那么我们在头文件中使用这三行,就相当于一个开关
比如在b.h中写上这三行。那么我们第一次展开a.h时候将b.h展开,然后就会判断是否定义 B_H 这个宏,然后定义,再执行代码。

#ifndef B_H 
#defined B_H
code...
#endif

下一次再展开b.h时候,就会再次判断,但这是我们已经定义过了,就不会再次定义,也不会再次执行我们的代码,也就是不会再次在预处理阶段展开。
也就是不会在#include "b.h" 处再次将这一行替换为头文件中代码。因为#include "b.h" 在预处理阶段会替换为头文件中代码。

总结

所以总结来说就是用条件编译定义一个宏(作为一个开关),去判断是否include当前头文件。以此避免重复定义!

http://www.lryc.cn/news/39318.html

相关文章:

  • 单片机能运行操作系统吗?
  • Python之webmagic爬虫优点与使用
  • 代码随想录动态规划 || 121 122
  • C++STL库中不可或缺的部分—string(模拟实现)
  • MySQL复合查询
  • PCIe 资料收集2
  • Linux网络编程(使用VScode远程登录ubuntu)
  • 如何提高项目估算精准度?关键看5大影响因子
  • 论文阅读笔记《Nctr: Neighborhood Consensus Transformer for Feature Matching》
  • 上位机系统Ubuntu 20.04与下位机arduino UNO通讯
  • hive面试题
  • 【CUDA】《CUDA编程:基础与实践》CUDA加速的关键因素
  • 数据结构【Golang实现】(四)——双向循环链表
  • 【Redis】高可用架构之哨兵模式 - Sentinel
  • 图片的美白与美化
  • 面试官:关于CPU你了解多少?
  • UI自动化测试-Selenium的使用
  • 嵌入式学习笔记——STM32的USART相关寄存器介绍及其配置
  • Android setContentView流程分析(一)
  • doris数据库操作数字遇到的问题
  • 3.13文件的IO操作
  • ffmpeg使用
  • spark中的并行度(分区数)/分区器如何确定
  • 00后女生“云摆摊”两周赚1.5万,实体店转战线上真的能赚钱吗?
  • 华为OD机试题 - 最优资源分配(JavaScript)| 机考必刷
  • 利用python判断字符串是否为回文
  • GDB 调用之ptype、set variable
  • 并发编程---阻塞队列(五)
  • 本科课程【计算机组成原理】实验1 - 输出ABCD程序的生成
  • Java并发编程(2) —— 线程创建的方式与原理