215

目的是std::make_pair什么?

为什么不直接做std::pair<int, char>(0, 'a')

这两种方法有什么区别吗?

4

7 回答 7

199

不同之处在于std::pair您需要指定两个元素的类型,而std::make_pair将创建一个具有传递给它的元素类型的对,而无需告诉它。无论如何,这就是我可以从各种文档中收集到的内容。

请参阅http://www.cplusplus.com/reference/std/utility/make_pair/中的此示例

pair <int,int> one;
pair <int,int> two;

one = make_pair (10,20);
two = make_pair (10.5,'A'); // ok: implicit conversion from pair<double,char>

除了它的隐式转换奖励之外,如果你不使用 make_pair 你必须做

one = pair<int,int>(10,20)

每次你分配给一个,随着时间的推移这会很烦人......

于 2012-02-14T01:39:57.687 回答
45

正如@MSalters 上面回答的那样,您现在可以在 C++11 中使用花括号来执行此操作(刚刚使用 C++11 编译器验证了这一点):

pair<int, int> p = {1, 2};
于 2014-02-24T18:28:47.870 回答
43

在 C++17 之前无法从构造函数中推断出类模板参数

在 C++17 之前,您不能编写如下内容:

std::pair p(1, 'a');

由于这会从构造函数参数推断模板类型,因此您必须将其显式编写为:

std::pair<int,char> p(1, 'a');

C++17 使这种语法成为可能,因此是make_pair多余的。

在 C++17 之前,std::make_pair允许我们编写不太冗长的代码:

MyLongClassName1 o1;
MyLongClassName2 o2;
auto p = std::make_pair(o1, o2);

而不是更冗长:

std::pair<MyLongClassName1,MyLongClassName2> p{o1, o2};

它重复类型,并且可以很长。

类型推断适用于 C++17 之前的情况,因为make_pair它不是构造函数。

make_pair本质上等同于:

template<class T1, class T2>
std::pair<T1, T2> my_make_pair(T1 t1, T2 t2) {
    return std::pair<T1, T2>(t1, t2);
}

同样的概念适用于insertervs insert_iterator

也可以看看:

最小的例子

为了使事情更具体,我们可以通过以下方式最低限度地观察问题:

主文件

template <class MyType>
struct MyClass {
    MyType i;
    MyClass(MyType i) : i(i) {}
};

template<class MyType>
MyClass<MyType> make_my_class(MyType i) {
    return MyClass<MyType>(i);
}

int main() {
    MyClass<int> my_class(1);
}

然后:

g++-8 -Wall -Wextra -Wpedantic -std=c++17 main.cpp

编译愉快,但是:

g++-8 -Wall -Wextra -Wpedantic -std=c++14 main.cpp

失败:

main.cpp: In function ‘int main()’:
main.cpp:13:13: error: missing template arguments before ‘my_class’
     MyClass my_class(1);
             ^~~~~~~~

而是需要工作:

MyClass<int> my_class(1);

或助手:

auto my_class = make_my_class(1);

它使用常规函数而不是构造函数。

差异为std::reference_wrapper

这条评论提到std::make_pair解包std::reference_wrapper,而构造函数没有,所以这是一个区别。待办事项示例。

GCC 8.1.0、Ubuntu 16.04测试。

于 2017-01-07T12:26:41.200 回答
26

make_pair使用和显式调用pair带有指定类型参数的构造函数之间没有区别。std::make_pair当类型很冗长时更方便,因为模板方法具有基于其给定参数的类型推导。例如,

std::vector< std::pair< std::vector<int>, std::vector<int> > > vecOfPair;
std::vector<int> emptyV;

// shorter
vecOfPair.push_back(std::make_pair(emptyV, emptyV));

 // longer
vecOfPair.push_back(std::pair< std::vector<int>, std::vector<int> >(emptyV, emptyV));
于 2012-02-14T01:49:24.283 回答
21

值得注意的是,这是 C++ 模板编程中的常见习语。它被称为对象生成器习语,您可以在此处找到更多信息和一个很好的示例。

编辑正如评论中有人建议的那样(已删除),以下是对链接的稍微修改的摘录,以防万一它中断。

对象生成器允许在不明确指定对象类型的情况下创建对象。它基于类模板所没有的函数模板的一个有用属性:函数模板的类型参数是从其实际参数自动推导出来的。是一个简单的例子,它根据函数的实际参数std::make_pair返回模板的一个实例。std::pairstd::make_pair

template <class T, class U>
std::pair <T, U> 
make_pair(T t, U u)
{
  return std::pair <T, U> (t,u);
}
于 2013-03-15T14:35:40.623 回答
5

make_pair 在直接构造函数上创建一个额外的副本。我总是对我的配对进行 typedef 以提供简单的语法。
这显示了差异(Rampal Chaudhary 的示例):

class Sample
{
    static int _noOfObjects;

    int _objectNo;
public:
    Sample() :
        _objectNo( _noOfObjects++ )
    {
        std::cout<<"Inside default constructor of object "<<_objectNo<<std::endl;
    }

    Sample( const Sample& sample) :
    _objectNo( _noOfObjects++ )
    {
        std::cout<<"Inside copy constructor of object "<<_objectNo<<std::endl;
    }

    ~Sample()
    {
        std::cout<<"Destroying object "<<_objectNo<<std::endl;
    }
};
int Sample::_noOfObjects = 0;


int main(int argc, char* argv[])
{
    Sample sample;
    std::map<int,Sample> map;

    map.insert( std::make_pair( 1, sample) );
    //map.insert( std::pair<int,Sample>( 1, sample) );
    return 0;
}
于 2013-11-28T13:40:41.543 回答
1

从 c++11 开始,只需对对使用统一初始化。所以而不是:

std::make_pair(1, 2);

或者

std::pair<int, int>(1, 2);

只需使用

{1, 2};
于 2019-03-14T10:04:46.233 回答