[C++再学习系列] Using声明和指令的工作原理

对于C++编译器,那么名字可见是至关重要的,太过的名字可见将导致名字查找效率的降低,而名字太少将导致无法找到所需类型或函数的名字,从而导致编译错误。除了最常用的include可以导入可见名字之外,using关键字也可以导入名字到特定的编译单元中(单个cpp文件)。

 

区别:

Using 声明: using namespace std;

Using 指令:  using N::Widget;

 

Using 用于导入特定名字空间下的名字实体,可以是全部名字实体(using namespace std;) ,也可以是特定的名字实体(using std::map;) 。

工作原理:using 声明将获取的是在遇到 using 声明 ( 指令 ) 瞬间所见到的名字空间中的实体。 这点是可以理解的,这样的工作方式有利于避免名字冲突( 过多导入污染) ,并加入名字查找过程。但这同时意味着:只有在using namespace std; 语句之前所include 的std 头文件的名字实体被引入,即using 并未引入所有来自std 的名字实体。

 

Using 关键字定理:绝对不要在 include 之前使用 using 声明或指令。

推论:不要在头文件中使用using 声明或指令,相反应该使用名字空间限定所有的名字,尤其是来自其他名字空间的名字。( 原因:头文件并不知道完全的include 信息,头文件总是被用于其他cpp 中,其后总会出现另外的include)

 

例子:

// snippet 1

namespace A {

int f(double);

}

 

// snippet 2

namespace B {

  using A::f;

  void g();

}

 

// snippet 3

namespace A {

int f(int);

}

 

// snippet 4

void B::g() {

  f(1);                         // which overload is called?

}

 

这几个文件先后出现顺序将决定f(1) 是否可编译( 所有的f 函数定义均不可见) ?以及重载哪个函数( 哪些函数可见,并参与重载) 。

错误地使用using 将导致名字空间污染,或错误地导入不完整的名字空间的瞬间快照( 导致无法找到特定名字实体) 。在头文件中,使用using 将使这个问题更加严重( 多处使用,使用顺序不可控) 。

 

例外:编写类成员级的using 声明以导入所需的基类成员名字是一个合法技巧,只要这样才能避免基类的名字被屏蔽,如重载函数不可见问题。(见派生类的函数重载)

 

posted on 2012-11-13 19:01  wencansz  阅读(280)  评论(0编辑  收藏  举报