【#ifndef, #define, 和 #endif】

前言

学习AFNetWoring源码的时候,在AFN的h借接口文件又看到了这几个宏定义,学习记录一下。
请添加图片描述

作用

#ifndef, #define, 和 #endif是C/CPP的预处理指令,常常用来条件编译和防止头文件重复包含。

简介

#ifndef 它是if not define的简写,是宏定义的一种,确切的说是预处理功能(宏定义、文件包含、条件编译)三种中的一种——条件编译。

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

  • #ifndef:如果未定义)用于判断指定的标识符是否已经被定义。如果该标识符未定义,则执行下面的代码块,否则跳过。
  • #define: 用于定义指定的标识符,通常用于定义宏。在条件编译中,一般用来定义一个标识符,以便在后面的代码中进行条件判断。
  • #endif :用于结束条件编译的代码块。

通常语法如下

#ifndef IDENTIFIER
#define IDENTIFIER

// 在这里放置需要执行的代码

#endif

其中,IDENTIFIER 是你自己定义的标识符名称,可以是任意合法的标识符。

这种结构的作用是为了防止头文件的重复包含。通过定义一个唯一的标识符,当该头文件第一次被包含时,IDENTIFIER 未定义,预处理器会执行 #define 下的代码块,标识该头文件已经被包含。当同一个头文件再次被包含时,由于 IDENTIFIER 已经被定义,预处理器会跳过 #define 下的代码块,从而避免了重复包含。

这种写法在头文件中很常见,以确保头文件的内容只会被编译一次,防止重复定义和冲突。

例子

以开头的代码为例

#ifndef _AFNETWORKING_
    #define _AFNETWORKING_

    #import "AFURLRequestSerialization.h"
    #import "AFURLResponseSerialization.h"
    #import "AFSecurityPolicy.h"

#if !TARGET_OS_WATCH
    #import "AFNetworkReachabilityManager.h"
#endif

    #import "AFURLSessionManager.h"
    #import "AFHTTPSessionManager.h"

#endif /* _AFNETWORKING_ */

这段代码的是OC的预编译指令,用于编译时对于代码进行条件判断和控制

  • #ifndef:预处理指令,用于判断一个标识符是否已经被定义过。如果标识符没有被定义过,则编译下面的代码块,否则忽略下面的代码块。
  • #define:预处理指令,用于定义一个标识符。在这个例子中,AFNETWORKING 就是被定义的标识符。
  • #endif:预处理指令,用于结束一个条件编译块。
    这段代码的作用是:当 AFNETWORKING 这个标识符没有被定义时,编译下面的代码块,并且将 AFNETWORKING 定义为已经被定义。这样可以避免头文件的重复引用和编译错误。

分解

#ifndef _AFNETWORKING_
#define _AFNETWORKING_

这里定义了一个标识符 _AFNETWORKING_,如果该标识符未定义(即未被之前的代码定义过),则执行下面的代码块,否则跳过。

#import "AFURLRequestSerialization.h"
#import "AFURLResponseSerialization.h"
#import "AFSecurityPolicy.h"

这里通过#import指令引入了一些其他的头文件,这些头文件可能包含了一些类或函数的声明。

#if !TARGET_OS_WATCH
#import "AFNetworkReachabilityManager.h"
#endif

这段代码使用了条件编译,判断是否为目标平台是不是watchOS,如果不是watchOS,则引入 "AFNetworkReachabilityManager.h" 头文件。

#import "AFURLSessionManager.h"
#import "AFHTTPSessionManager.h"

这里又引入了两个头文件 "AFURLSessionManager.h""AFHTTPSessionManager.h"

最后,代码块的结束部分是 #endif,用于结束条件编译的代码块。

整个结构的目的是为了确保头文件的内容只会被编译一次,避免了重复定义和冲突。

  • 7
    点赞
  • 48
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
#ifndef,#define和#endifC++中用于避免头文件重复引用的预处理指令。它们一般是成对出现的。当一个头文件被多个文件引用时,为了防止重复定义和冲突,可以使用这些指令。 #ifndef是一个条件编译指令,用于判断某个标识符是否已经定义。如果这个标识符未定义,那么条件成立,可以继续执行下面的代码。如果已经定义了这个标识符,那么条件不成立,下面的代码将被跳过。 #define是用于定义一个标识符,并给定一个值。一般情况下,我们使用它来定义一个宏。在这个宏定义中,我们可以将标识符定义为一个值,或者直接定义为一段代码。 #endif是用于结束条件编译块的指令。它与#ifndef配合使用,用于标记条件编译的结束点。 举个例子,首先我们可以在头文件中使用#ifndef来判断某个标识符是否已经定义,如果没有定义,就使用#define来定义这个标识符,并编写相应的代码。然后,在其他的源文件中,我们可以使用#include来包含这个头文件。当源文件被编译时,预处理器会根据#ifndef和#define的判断来决定是否编译这部分代码。如果这个头文件已经被引用过了,预处理器会跳过这部分代码,从而避免重复定义和冲突的问题。 总的来说,使用#ifndef,#define和#endif指令可以有效地避免头文件的重复引用问题,确保代码的正确性和可维护性。需要注意的是,并不是所有的头文件都需要加入这些指令,只有在可能被多个文件引用的情况下才需要使用。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [C++ 头文件 #ifndef #define #endif 作用及用法](https://blog.csdn.net/qq_37581730/article/details/127770758)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [#ifndef/#define/#endif使用详解](https://blog.csdn.net/m0_48990191/article/details/115007141)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值