提问人:Robert 提问时间:8/28/2020 最后编辑:Robert 更新时间:8/28/2020 访问量:1184
库和命名空间之间有什么关系?
What's the relation between libraries and namespaces?
问:
刚开始拼贴,我是编程界的新手。所以,当我学习C++时,我遇到了一个不允许我的问题:“如果我已经有iostream,为什么我需要在我的代码中包含”使用命名空间std“才能编写或读取?”,因为我被告知“cin/cout”已经在iostream库中defind,但我注意到如果我单独编写其中一行,它将给出编译错误。因此,iostream 和“std”命名空间之间有什么关系......有人可以解释一下吗?谢谢!<3
答:
#include 与使用
简单来说:
#include <iostream> // include the header
int main() {
// now you can use stuff declared in that header
std::cout << "Hello world" << std::endl;
// if you are lazy you can "use" things:
using std::cout;
using std::endl;
cout << "Hello World" << endl;
}
你不必写!这样做的情况非常罕见,而真正造成巨大伤害的情况非常频繁,根据经验,您可以记住:永远不要使用它!有关详细信息,请参阅此处:为什么“使用命名空间 std;”被认为是不良做法?。必须认识到,完全限定名称和不完全限定名称之间的区别不仅仅是或多或少地键入 5 个字符(请继续阅读......using namespace std;
std::cout
cout
库与命名空间
库和命名空间之间有什么关系?
标准库将所有内容都放在命名空间中。命名空间有助于将事物分开。不同的库可以包含一个,并且不会混淆,因为我们有命名空间。std
other_namespace::vector
std::vector
真正酷的东西
使用命名空间的一个更深层次的原因是参数相关查找。我将尝试用一个简单的例子来解释。假设您正在使用一个带有某个函数模板的库,该函数模板对您必须提供的类型的对象执行某些操作:
namespace library {
template<typename T>
void do_something(T& a,T& b){
std::cout << "wrong...\n";
std::swap(a,b); // (1)
std::cout << "correct\n";
using std::swap;
swap(a,b); // (2)
}
}
我拿了两个对象,交换了两次。你必须忍受我一秒钟才能理解为什么 (1) 是错误的,只有 (2) 是正确的。现在我们有一个库函数模板,要使用它,我们需要一些类型:T
namespace A {
struct foo{};
void swap(foo& a,foo& b) {
std::cout << "A::swap" << "\n";
}
}
想象是这样的,我们知道比实例更好的方法。实际上是空的,所以对于两个对象,我们什么都不做。foo
std::swap
swap
foo
swap
让我们回顾一下:标准库附带了。有人写了一个我们想要使用的库(称为)。我们希望库代码调用而不是 .库作者甚至不知道它的存在。std::swap
library
A::swap
std::swap
A::swap
连同上面的 和 ,这段代码A
library
int main() {
A::foo a,b;
library::do_something(a,b);
}
将打印:
wrong...
correct
A::swap
现场示例。发生了什么事?这一行:
std::swap(a,b); // (1)
电话,毫无疑问。不是我们想要的。我们希望库代码调用我们的 .std::swap
A::swap
现在这个:
using std::swap;
swap(a,b); // (2)
第一行将名称从函数的作用域中提取出来。在第二行中,ADL 终于开始发挥作用,因为它说没有 .简而言之,ADL 是: 和 来自命名空间 ,因此当编译器搜索所有可能的 s 时,它也会搜索 。如果它找到一个,那么它就会调用它(如果它没有找到一个,那么它仍然来自)。因此,只有 (2) 调用我们的自定义交换。swap
std
swap
std::swap
a
b
A
swap
A
A
A
swap
std
这仅适用于命名空间。“很酷的东西”是库作者不需要知道你的命名空间的任何信息,但他们的库代码仍然会从你的命名空间调用你的函数(如果它存在)。
我应该注意,并非所有代码都是通用库代码。通常,您希望在编写代码时知道每个细节中发生了什么,您想知道调用了哪些函数。通常,您不希望代码的行为因是否包含特定标头而有所不同。因此,使用完全限定的函数调用来最好使用大量代码: 。std::foo
结论
我希望我能说服你,命名空间不仅仅是或多或少地键入一些字符。 因为懒惰完全错过了命名空间的重点。另一方面,通过以下方式将名称拉入范围
完全没问题并启用 ADL。using namespace std;
using std::foo; foo();
评论
using namespace std;
using namespace std;
using namespace std
IOSTREAM 是一个库。这是有人为你编写的代码,所以你不必这样做。通过添加,您可以告诉预处理器粘贴该代码。但是,此代码提供的函数和结构的名称可能会干扰其他函数和结构。但这不是问题,因为你可以通过将它们放在命名空间中来分隔它们,STL(上游是其中的一部分)使用(标准的缩写,发音为“standod”)来做到这一点。当某个内容位于命名空间中时,您必须命名该命名空间才能访问其中的内容。即 .但有时您不希望每次都想从 STL 访问某些内容时都进行编写。这就是对你的作用。这样,您只需键入 .但这是一个非常糟糕的主意!#include <iostream>
std
std::cout
std::
using namespace std
cout
评论
using namespace std;
cout
roberts::cout
std::cout
cout
cout
std
库和命名空间按约定相关。
按照惯例,库提供给程序员-用户的符号包含在命名空间中。这样可以组织事物,并且有一些更高级别的语言功能 (ADL) 意味着命名空间中的代码的行为与命名空间外的代码不同。
当你输入时,你告诉编译器“当你遇到一个符号时,也要查看一下,看看你是否能确定它是什么”。在“文件”范围内执行此操作通常是一个非常糟糕的主意;在单个简短的函数中执行此操作是可用的,但超过此范围可能会导致非常棘手的错误。using namespace std;
std
标准的、专业的交互方式是用命名空间作为符号的前缀:namespace std
std::cout << "Hello world\n";
而不是
using namespace std;
cout << "Hello world\n";
绝对不会:
using namespace std;
int main() {
cout << "Hello world\n";
}
您还可以获取单个符号,这并不像导入整个命名空间那么糟糕:
using std::cout;
cout << "Hello world\n";
但也应避免在“文件”范围内使用。
#include <iostream>
这包括从系统搜索路径命名的头文件。 是标准库的一部分。按照约定(和 C++ 标准),为程序提供的符号位于 中。iostream
iostream
iostream
namespace std
通过在命名空间中放置符号,可以避免与代码发生冲突。中有很多很多符号,如果将一些未知数量的符号推入全局命名空间,则很容易出现错误或以意想不到的方式调用错误的函数。std
#include <iostream>
std::cout
和 和 都是告诉编译器在哪个命名空间中查找符号的方法。using namespace std; cout
using std::cout
cout
#include <iostream>
包括 ;没有它,你的代码就不知道它的存在。cout
namespace std
C++ 是从 C 发展而来的,C 具有文本包含模型。 实际上获取文件的内容并将其复制/粘贴到您的文件中。然后,编译器读取该扩展文件,并在 中查找符号。#include
iostream
<iostream>
因为这种文本包含可能会推很多东西,所以将其隔离到 a 可以防止您(程序员)出现问题。namespace
最近,C++ 添加了模块。模块是指令的替代方法,因为它直接从库中获取符号并将其注入到代码中,而无需大量复制粘贴。#include
在模块中,命名空间仍未直接连接到模块。您可以
import std;
或
import std.iostream;
这只会将仍在 中的库符号导入到您的代码中。(C++标准增加了模块,但还没有模块化std库,所以上面的那些名字都是推测)。std
namespace std
符号查找不直接连接到符号导入。
这样可以大块地导入符号,同时更仔细地进行查找。
评论
cpp
namespace bob { struct hello {}; }
namespace alice { struct world {}; }
bob::hello
alice::world
图书馆
库有预先编写的部分代码,以便为您提供功能。可以是函数/重载运算符等的形式。
有两种类型的库:
标准库,例如 库的名称用尖括号括起来。
#include <iostream>
用户定义/制造,例如 库的名称用双引号括起来。
#include "randomLib.h"
命名空间
当您的项目需要多个库时。两者都可能包含多个具有相同名称的方法(函数定义),或者单个库可能使用相同的函数名称,但位于不同的命名空间中。命名空间用于消除编译器和用户的混淆或歧义。
- 假设 lib 1 有,lib 2 有
namespace abc{ foo(); }
namespace def{ foo(); }
因此,您将为您做或为您所需的功能。这里 abc/def 是 ,称为作用域解析运算符,是您要调用的方法。abc::foo()
def::foo()
namespace
::
foo()
评论
std
#include <iostream> int main() { std::cout << "Hello world\n"; }
-看。不。using namespace ..