现在的位置: 首页 > 综合 > 正文

#pragma once与 #ifndef的区别

2013年10月07日 ⁄ 综合 ⁄ 共 1138字 ⁄ 字号 评论关闭

 原文URL:http://www.cppblog.com/szhoftuncun/archive/2007/10/28/35356.html

为了避免同一个文件被include多次

1   #ifndef方式
2   #pragma once方式

在能够支持这两种方式的编译器上,二者并没有太大的区别,但是两者仍然还是有一些细微的区别。
    方式一:

    #ifndef __SOMEFILE_H__
    #define __SOMEFILE_H__
    ... ... // 一些声明语句
    #endif

    方式二:

    #pragma once
    ... ... // 一些声明语句

    #ifndef的方式依赖于宏名字不能冲突,这不光可以保证同一个文件不会被包含多次,也能保证内容完全相同的两个文件不会被不小心同时包含。当然,缺点就是如果不同头文件的宏名不小心“撞车”,可能就会导致头文件明明存在,编译器却硬说找不到声明的状况。

    #pragma once则由编译器提供保证:同一个文件不会被包含多次。注意这里所说的“同一个文件”是指物理上的一个文件,而不是指内容相同的两个文件。带来的好处是,你不必再费劲想个宏名了,当然也就不会出现宏名碰撞引发的奇怪问题。对应的缺点就是如果某个头文件有多份拷贝,本方法不能保证他们不被重复包含。当然,相比宏名碰撞引发的“找不到声明”的问题,重复包含更容易被发现并修正。

   方式一由语言支持所以移植性好,方式二 可以避免名字冲突

补充一下:通过 http://en.wikipedia.org/wiki/Pragma_oncehttp://en.wikipedia.org/wiki/Include_guard 获取更多关于它们的信息。

1. #pragma once是非标准的。
2. #pragma once是一种高阶的机制,能加快编译速度。
3. 部份编译器,例如GCC,会有一些特殊的处理去优化include,但是,对于#pragma once,却不一定会有优化代码对其优化。
4. #pragma once并不是一种完美的机制,它是一种编译器所做的处理,它依赖于编译器是否能够正确的工作。
5. 如果你想你的代码具有可移植性,能够在各种编译器上跑,可是你不知道编译器有没有实现#pragma once,但又想使用#pragma once去加速代码编译(没准人家就实现了呢),怎么做?那你就把#pragma once和#ifndef统统加上吧。

#pragma once
#ifndef GRANDFATHER_H
#define GRANDFATHER_H
 
struct foo
{
    int member;
};
 
#endif /* GRANDFATHER_H */

抱歉!评论已关闭.