C的开发人员会经常使用#define,即用宏来声明常量,但宏却是全局的,对大的工程很难维护,经常是导致名字冲突。还好,C++给我们带来了namespace名字空间。它的使用如下,名字空间可以把一组逻辑分组,同时名字空间也是一种作用域。
namespace outspname { const int CVAR1 = 1; const char* const CVAR2 = "33333"; void test(); namespace inspname { enum { A, B, C}; class Klass { }; } }
但即使一个简单的名字空间,其中也有不少的玄机。
1.当某个名字在自己的空间之外使用,在反复地在前面加上名字空间作为限定词, 如
const int local = outspname::inspname::A
这样写是不是很令人烦。在某个小的局部作用域内,我们可以通过一个使用声明。如
{ using outspname::inspname::A; const int local = A; }
2.还有,我们也可以通过一个使用指令把该名字空间下所有的名字变成可用。如下所示,与第一点的用法区别,是using 后面有个namespace。同样只在转换时,或者在一个小的局部作用域内使用using namesapce,否则也会带来名字的污染。
{ using namespace outspname; const int local2 = CVAR1; const int local2 = inspname::B; { using namespace inspname; Klass* p = new Klass(); } }
但使用using namespace这种用法时,要注意下面一点,如在某个.h中声明了有testname::test的方法。
namespace testname { void test(int param); }
在其.cpp中,不能使用如下这种方式,test方法只是此编译单元的一个局部方法,并非testname名字空间的test方法实现。
using testname; void test(int param) { }
正确的使用方式是
namespace testname { void test(int param) { } }
或者是
void testname::test(int param) { }
3.名字空间的别名,当名字空间很长或嵌套很深时,我们可以使用名字空间别名,用法如下:
namespace oin = outspname::inspname;
4.无名名字空间,无名名字空间主要是保持代码的局部性,使用如下:
namespace { const int CVAR1 = 1; void test(); }
但一定要注意的一点是,在C++编译器实现时,无名名字空间其实是有名字的,这个隐含的名字跟它所在编译单元名字相关。所以基于这一点,我们不能跨编译单元使用无名名字空间中的名字。
上面的声明等价于
namespace $$$ { const int CVAR1 = 1; void test(); } using namespace $$$;
其中$$$在其所在的作用域里具有惟一性的名字,每个编译单元里的无名名字空间也是互不相同的,using namesapce $$$只是当前的编译单元的隐含名字,所以不能跨编译单元使用无名名字空间中的名字。
假设上面的test方法在是a.h与a.cpp中定义与实现的,但在b.h或b.cpp中就不能直接使用test方法或CVAR1。因为在b的这个编译单元中链接的是b这个编译单元中的test符号,并非a编译单元中的test符号,也就会出现未定符号。
5.要避免名字空间使用很短的名字,也不能太长,更不能嵌套太深了,个人觉得不要超过4层。