我试图std::pair用这种风格制作:
#include <iostream>
struct A {
A() noexcept {
std::cout << "Created\n";
}
A(const A&) noexcept {
std::cout << "Copy\n";
}
A(A&&) noexcept {
std::cout << "Move\n";
}
};
int main() {
std::pair<A, A> a{ {},{} };
return 0;
}
Run Code Online (Sandbox Code Playgroud)
并得到这样的输出:
Created
Created
Copy
Copy
Run Code Online (Sandbox Code Playgroud)
代替
Created
Created
Move
Move
Run Code Online (Sandbox Code Playgroud)
但是如果我定义我的匿名对象类型(例如std::pair<A, A> a{A{}, A{}})或使用std::make_pair<A, A>({}, {})我得到正确的结果。
std::pair构造函数必须使用std::forward<U1>和std::forward<U2>来初始化对象,因此我认为我的配对使用了错误的构造函数。为什么?
Hol*_*Cat 55
std::pair定义方式有问题。我什至会说这是标准中的一个小缺陷。
它有两个可以在这里使用的构造函数:
pair(const T1 &x, const T2 &y);, 其中T1,T2是 的模板参数pair。
template <class U1, class U2> pair(U1 &&x, U2 &&y);
如果这样做std::pair<A, A> a{A{}, A{}});,则选择第二个构造函数并且一切正常。
但是如果你这样做了std::pair<A, A> a{{}, {}};,编译器就不能使用第二个构造函数,因为它不能推导出U1, U2,因为{}它本身没有类型。所以第一个构造函数被使用,你得到一个副本。
为了让它正常工作,std::pair应该有一个额外的非模板构造函数pair(T1 &&, T2 &&),并且有两个额外的构造函数:pair(const T1 &, T2 &&)和pair(T1 &&, const T2 &)。