zoukankan      html  css  js  c++  java
  • #ifdef 和#endif的作用:防止头文件循环引用

    iOS的pch文件中常见的代码解释:

    一般情况下,源程序中所有的行都参加编译。但是有时希望对其中一部分内容只在满足一定条件才进行编译,也就是对一部分内容指定编译的条件,这就是“条件编译”。

    有时,希望当满足某条件时对一组语句进行编译,而当条件不满足时则编译另一组语句。 

    条件编译命令最常见的形式为: 

    #ifdef 标识符 
    程序段1 
    #else 
    程序段2 
    #endif

    它的作用是:当标识符已经被定义过(一般是用#define命令定义),则对程序段1进行编译,否则编译程序段2。 
    其中#else部分也可以没有,即: 
    #ifdef 
    程序段1 
    #denif

    在头文件中使用#ifdef和#ifndef是非常重要的,可以防止双重定义的错误。

    如你在头文件aaa.h中定义了一个类aaa如下:   
      class   aaa   
      {   
      };   
      如果两次#include   "aaa.h"(不见得是直接,也有可能两个不同的头文件中都包含了这个头文件)就会出错,因为相同的类不能定义两次。把aaa.h稍做修改:   
      #ifndef   _aaa_   
      #define   _aaa_   
      class   aaa   
      {   
      };   
      #endif   
      就可以避免这样的问题。

    因为当你已经包含过这个文件,_aaa_就会有了定义,那么#ifndef的条件为假,就不会再执行后面的类定义了。 

    #ifdef和#endif必须成对使用。   
      从理论上讲可以出现在任何地方(头文件和实现文件中)   
      通常为了防止头文件被多次包含,在头文件中使用是必须的:   
      如:#ifndef   MY_HEAD_H   //头文件开头,名字是任意的,注意不要和其它头文件冲突        
        
      头文件声明   
          #endif     //头文件结尾

    有时候,在b.h中会include   "a.h"  ,在"c.h"中会include   "b.h"及include"a.h", 这时,如果不用ifndef/endif,就会包含两次a.h,产生错误。 

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

    #ifndef <标识> 
    #define <标识>

    ...... 
    ...... 

    #endif

    <标识>在理论上来说可以是自由命名的,但每个头文件的这个“标识”都应该是唯一的。

    命名规范:

    1.标识的命名规则一般是头文件名全大写,前后加下划线,并把文件名中的“.”也变成下划线,如:stdio.h 

    #ifndef _STDIO_H_ 
    #define _STDIO_H_

    ......

    #endif 

    2.在#ifndef中定义变量出现的问题(一般不定义在#ifndef中)。

    #ifndef AAA
    #define AAA
    ...
    int i;
    ...
    #endif
    里面有一个变量定义,在vc中链接时就出现了重复定义的错误,而在c中成功编译。

        想必很多人都看过“头文件中的 #ifndef/#define/#endif 防止该头文件被重复引用”。但是是否能理解“被重复引用”是什么意思?是不能在不同的两个文件中使用include来包含这个头文件吗?如果头文件被重复引用了,会产生什么后果?是不是所有的头文件中都要加入#ifndef/#define/#endif 这些代码?

     

        其实“被重复引用”是指一个头文件在同一个cpp文件中被include了多次,这种错误常常是由于include嵌套造成的。比如:存在a.h文件#include "c.h"而此时b.cpp文件导入了#include "a.h" 和#include "c.h"此时就会造成c.h重复引用。

     

    头文件被重复引用引起的后果:

    有些头文件重复引用只是增加了编译工作的工作量,不会引起太大的问题,仅仅是编译效率低一些,但是对于大工程而言编译效率低下那将是一件多么痛苦的事情。
    有些头文件重复包含,会引起错误,比如在头文件中定义了全局变量(虽然这种方式不被推荐,但确实是C规范允许的)这种会引起重复定义。

     

        是不是所有的头文件中都要加入#ifndef/#define/#endif 这些代码?

        答案:不是一定要加,但是不管怎样,用ifnde xxx #define xxx#endif或者其他方式避免头文件重复包含,只有好处没有坏处。个人觉得培养一个好的编程习惯是学习编程的一个重要分支。

     

        下面给一个#ifndef/#define/#endif的格式:

        #ifndef A_H意思是"if not define a.h"  如果不存在a.h

        接着的语句应该#define A_H  就引入a.h

        最后一句应该写#endif   否则不需要引入

    --------------------------------------------------------------------------------------------------
    #ifndef GRAPHICS_H // 防止graphics.h被重复引用 
    #define GRAPHICS_H 


    #include <math.h> // 引用标准库的头文件 
    … 
    #include “header.h” // 引用非标准库的头文件 
    … 
    void Function1(…); // 全局函数声明 
    … 
    class Box // 类结构声明 

    … 
    }; 
    #endif
    --------------------------------------------------------------------------------------------------

  • 相关阅读:
    [转]百度地图API详解之地图坐标系统
    [转]MBTiles 离线地图演示
    [shell 编程] if [ $# -eq 0 ]该语句是什么含义?
    [shell编程] sh脚本异常:/bin/sh^M:bad interpreter: No such file or directory
    [Android Studio] Android studio 多渠道打包(超简洁版)
    [Android Pro] Android中全局Application的onCreate多次调用问题
    [Java基础] Java如何实现条件编译
    [Git] 根据commiter过滤该用户的所有提交
    [Android Pro] Notification的使用
    [Git] Git把Tag推送到远程仓库
  • 原文地址:https://www.cnblogs.com/wangbinios/p/7573841.html
Copyright © 2011-2022 走看看