现代程序猿们对Namespace(命名空间)的使用已经习以为常了,而且现在的编译器对Namespace肯定都有比较好的支持。但是有没有想过在很遥远的某个年代,当时编译器并不支持Namespace?这种情况当然是存在的,所以为了保持向后兼容性,STL中使用了条件编译。

最近读STL源码(SGI版本)的时候发现了一种很奇怪的情况:有些头文件中包含了“__STL_BEGIN_NAMESPACE”和“__STL_END_NAMESPACE”这样不伦不类的语句,而且编辑器在下面加上了下划线,显然是编辑器所不能解析的。

在网上搜索了一下,发现原因如下:

#ifndef __SGI_STL_INTERNAL_ITERATOR_H
#define __SGI_STL_INTERNAL_ITERATOR_H

 __STL_BEGIN_NAMESPACE

struct input_iterator_tag {};
。。。。。。。。。。。。
这段代码来自sgi stl中 stl_iterator.h
请问各位高手,这段代码中的__STL_BEGIN_NAMESPACE是干什么用的,放在这里不会出错吗?

 

答案1:

看stl_config.h:
00205 # if defined(__STL_USE_NAMESPACES) && !defined(__STL_NO_NAMESPACES)
00206 #   define __STD std
00207 #   define __STL_BEGIN_NAMESPACE namespace std {
00208 #   define __STL_END_NAMESPACE }
00209 #   define  __STL_USE_NAMESPACE_FOR_RELOPS
00210 #   define __STL_BEGIN_RELOPS_NAMESPACE namespace std {
00211 #   define __STL_END_RELOPS_NAMESPACE }
00212 #   define __STD_RELOPS std
00213 # else
00214 #   define __STD
00215 #   define __STL_BEGIN_NAMESPACE
00216 #   define __STL_END_NAMESPACE
00217 #   undef  __STL_USE_NAMESPACE_FOR_RELOPS
00218 #   define __STL_BEGIN_RELOPS_NAMESPACE
00219 #   define __STL_END_RELOPS_NAMESPACE
00220 #   define __STD_RELOPS
00221 # endif

在支持namespace的环境下,配合__STL_END_NAMESPACE使用namespace std
namespace std{ //__STL_BEGIN_NAMESPACE
...
} //__STL_END_NAMESPACE

答案2:

__STL_BEGIN_NAMESPACE宏是在某个配置文件中定义的,就sgi来说,此宏为了兼容一些早期代码,允许stl模板库不是用std命名空间包裹,__STL_BEGIN_NAMESPACE根据用户配置,被定义为“空”或者“namespace std {”之类的实际代码。

 

我查看了一下,在STL所有底层文件中(以"stl_"开始的头文件)几乎都含有这两个宏,文件开始时会包含“__STL_BEGIN_NAMESPACE”,结尾处会包含“__STL_END_NAMESPACE”。这样,在支持Namespace的用户环境下会被转换成 namespace std { },在不支持Namespace的用户环境下会被忽略。

posted on 2013-05-26 11:27  wanghetao  阅读(1633)  评论(1编辑  收藏  举报