以前用C#.NET的时候,没有深究过namespace的用法,最近看C++ Primer中关于namespace的讲解有自己的一点体会:
首先,来看我写的一小段代码:
#include <iostream>
using namespace std;
void doSomething()
{
cout << "doSomething() called" << endl;
}
namespace first
{
void doSomething()
{
cout << "first::second::third::doSomething() called" << endl;
}
namespace second
{
void doSomething()
{
cout << "first::second::doSomething() called" << endl;
}
namespace third
{
void doSomething()
{
cout << "first::second::third::doSomething() called" << endl;
}
}
}
namespace test
{
namespace testInner
{
void func()
{
using namespace first::second::third;
doSomething();
}
}
}
}
int main()
{
test::testInner::func();
return 0;
}
粗看起来,似乎没什么错误,可是编译未能通过,报错:error C2668: 'doSomething' : ambiguous call to overloaded function Error executing cl.exe.
究其原因是因为在函数func()中,语句using namespace first::second::third将命名空间first::second::third下的所有成员定义为first下的的成员,而first::second::third::doSomething()与first::doSomething()重复定义(函数签名相同,未能构成重载)。
为了使程序能编译通过,解决方法之一就是将first::doSomething()删除,比如:
//...原程序不变
namespace first
{
// void doSomething() // 删除此函数
// {
// cout << "first::second::third::doSomething() called" << endl;
// }
namespace second
{
//...main()函数不变
此时再次编译即可通过。那么现在运行结果是什么呢?前面说到,语句using namespace first::second::third将命名空间first::second::third下的所有成员声明为first下的的成员,因而此时first::second::third::doSomething()应覆盖::doSomething(),故结果输出:
first::second::third::doSomething() called
小结:
在如下代码中,using指示符将命名空间namespaceHome::namespace1::...::namespaceN下的成员声明为namespaceHome下的成员,即using指示符所在的命名空间(namespaceHome::namespaceU1::..namespaceUM)与using指示符所声明的命名空间(namespaceHome::namespace1::...::namespaceN)所在的第一个母命名空间下的成员。
namespaceHome
{
namespace1
{
... // N层命名空间
{
namespaceN
{
... // 各成员
}
}
}
namespaceU1
{
... // M层命名空间
{
namespaceUM
{
using namespace1::...::namespaceN;
}
}
}
}