我有Java背景,使用的是包,而不是名称空间。我习惯于将一起工作以形成完整对象的类放入包中,然后从该包中重用它们。但是现在我在用c++工作。

在c++中如何使用名称空间?您是为整个应用程序创建单个名称空间,还是为主要组件创建名称空间?如果是,如何从其他名称空间中的类创建对象?


当前回答

为了避免赘述,Mark Ingram已经说了一个使用名称空间的小技巧:

避免在头文件中使用“using namespace”指令——这会为导入该头文件的程序的所有部分打开命名空间。在实现文件(*.cpp)中,这通常不是大问题——尽管我更喜欢在函数级别上使用“using namespace”指令。

我认为命名空间主要用于避免命名冲突,而不一定用于组织代码结构。我主要用头文件/文件结构来组织c++程序。

有时在较大的c++项目中使用名称空间来隐藏实现细节。

using指令的附加说明: 有些人喜欢只对单个元素使用“using”:

using std::cout;  
using std::endl;

其他回答

我使用c++命名空间的方式与我在c#、Perl等中使用的方式相同。它只是标准库内容、第三方内容和我自己的代码之间的符号语义分离。我会将自己的应用程序放在一个名称空间中,然后将可重用的库组件放在另一个名称空间中进行分离。

在Java中:

package somepackage;
class SomeClass {}

在c++中:

namespace somenamespace {
    class SomeClass {}
}

使用它们,Java:

import somepackage;

和c++:

using namespace somenamespace;

此外,全名是“somepackage”。用于Java的SomeClass和用于c++的somenamespace::SomeClass。使用这些约定,您可以像在Java中一样进行组织,包括为名称空间创建匹配的文件夹名称。但是没有文件夹->包和文件->类的要求,因此您可以根据包和名称空间独立地命名文件夹和类。

不要相信每个人都说名称空间只是名称空间。

它们很重要,因为它们被编译器用来应用接口原则。基本上,可以用一个例子来解释:

namespace ns {

class A
{
};

void print(A a)
{
}

}

如果你想打印一个A对象,代码将是这样的:

ns::A a;
print(a);

注意,在调用函数时没有显式地提到名称空间。这是接口原则:c++将一个以类型作为参数的函数视为该类型接口的一部分,因此不需要指定命名空间,因为形参已经隐含了命名空间。

为什么这个原则很重要呢?假设类A的作者没有为这个类提供print()函数。你必须自己提供一份。作为一名优秀的程序员,您将在自己的名称空间中定义这个函数,或者可能在全局名称空间中定义。

namespace ns {

class A
{
};

}

void print(A a)
{
}

你的代码可以在任何你想要的地方调用print(a)函数。现在想象一下,多年以后,作者决定提供一个print()函数,它比你的更好,因为他知道他的类的内部结构,并且可以提供一个比你的更好的版本。

然后,c++作者决定使用他的print()函数版本,而不是在另一个名称空间中提供的函数,以尊重接口原则。print()函数的“升级”应该尽可能简单,这意味着您不必更改对print()函数的每次调用。这就是为什么在c++中不需要指定命名空间就可以调用“接口函数”(与类在同一命名空间中的函数)。

这就是为什么在使用c++名称空间时应该将其视为“接口”,并记住接口原则。

如果你想要更好地解释这种行为,你可以参考Herb Sutter的《例外c++》这本书

@marius

是的,你可以同时使用几个命名空间,例如:

using namespace boost;   
using namespace std;  

shared_ptr<int> p(new int(1));   // shared_ptr belongs to boost   
cout << "cout belongs to std::" << endl;   // cout and endl are in std

[2014年2月-(真的有那么长时间吗?):正如乔伊在下面指出的,这个特殊的例子现在是模棱两可的。Boost和std::现在各有一个shared_ptr.]

一般来说,如果我认为一段代码可能与其他库有函数或类型名称冲突,我会为它创建一个名称空间。它也有助于品牌代码,ala boost::。