C++ map.insert 传参类型不同,构造/析构次数不同
1. 传参方式
使用 insert 为 map 插值时,insert 的传参包含以下几种可能:
- make_pair 生成对象
- pair(key_type, value_type) 生成对象
- pair(const key_type, value_type) 生成对象
- map<key_type, value_type>::value_type 生成对象
不同的传参,导致不同次数的构造 / 析构函数调用。
2. 测试代码与结果
/*
在 map 中不同的 insert 传参类型,导致的不同次数的构造/析构函数调用。
*/ #include <iostream>
#include <vector>
#include <map>
#include <string> using namespace std; class Data
{
public:
Data(void)
{
cout << "Constructor" << endl;
} Data(const Data& other)
{
cout << "Copy Constructor" << endl;
} Data& operator=(const Data& other)
{
cout << "Assignment operator" << endl;
return *this;
} ~Data(void)
{
cout << "Destructor" << endl;
}
}; class Tracker
{
public:
Tracker(string name)
{
_name = name;
cout << "### Testing " << name << endl;
} ~Tracker(void)
{
cout << "### Done" << endl << endl;
} private:
string _name;
}; int main () {
{
Tracker tracker("insert data using make_pair");
map<int, Data> c;
Data d = Data();
cout << "---- Begin ---- " << endl;
c.insert(make_pair(, d));
cout << "----- End -----" << endl;
} {
Tracker tracker("insert data using pair<key_type, value_type>");
map<int, Data> c;
Data d = Data();
cout << "---- Begin ---- " << endl;
c.insert(pair<int, Data>(, d));
cout << "----- End -----" << endl;
} {
Tracker tracker("insert data using pair<const key_type, value_type>");
map<int, Data> c;
Data d = Data();
cout << "---- Begin ---- " << endl;
c.insert(pair<const int, Data>(, d));
cout << "----- End -----" << endl;
} {
Tracker tracker("insert data using map<key_type, value_type>::value_type");
map<int, Data> c;
Data d = Data();
cout << "---- Begin ---- " << endl;
c.insert(map<int, Data>::value_type(, d));
cout << "----- End -----" << endl;
} return ;
}

3. 调用流程分析
3.1 value_type 2 次 Copy Constructor 产生 1 个临时变量
分解后的 value_type 调用细节对比图

- 构造 value_type 的临时变量,一次 Copy Constructor
- insert value_type 类型的参数,一次 Copy Constructor
- 析构 value_type 的临时变量,一次 Destructor
3.2 pair(const key_type, value_type) 与 value_type 过程相同
原因:value_type 的定义是: pair<const key_type,mapped_type>
参见:http://www.cplusplus.com/reference/map/map/
3.3 pair(key_type, value_type) 3 次 Copy Constructor 产生 2 个临时变量
根据下图的流程拆解可以看出,非 const 的 key_type 导致 insert 过程增加了一次 Copy Constructor

3.4 make_pair 4 次 Copy Constructor 产生 3 个临时变量
新增的一次 Copy Constructor 来自于 make_pair 过程。
make_pair 的过程是:Constructs a pair object with its first element set to x and its second element set to y.
伪代码如下:
template <class T1,class T2>
pair<T1,T2> make_pair (T1 x, T2 y)
{
return ( pair<T1,T2>(x,y) );
}
参见:http://www.cplusplus.com/reference/utility/make_pair/?kw=make_pair
make_pair 时的 key 使用 const 变量,并未能减少 insert 阶段的一次 Copy Constructor
const int k = ;
c.insert(make_pair(k, d));
4. 结果分析
- insert 时,如果 key_type 不是 const 类型, 则增加一次 Copy Constructor
- make_pair 无法产生 const key_type 的 pair,导致 insert 增加一次 Copy Constructor
- map<key, value>::value_type 的 key 是 const 类型,使用简单,性能好。
C++ map.insert 传参类型不同,构造/析构次数不同的更多相关文章
- LiveBOS Webservice传参类型为list数组
昨天有使用soap传输数据到Webservice,其中字符串类型的都已经传输成功,但是有几个参数传输失败,java服务器端收到的空值. 因为我是php的,然后接收端是java制作的,其中有几个参数是l ...
- httpclient传参类型与响应参数接收
https://blog.csdn.net/qq_26562641/article/details/72817457 https://blog.csdn.net/chenjf0221/article/ ...
- C++ 传参时传内置类型时用传值(pass by value)方式效率较高
来源:唐磊的个人博客<C++ 传参时传内置类型时用传值(pass by value)方式效率较高> 在<Effective C++>里提到对内置(C-like)类型在函数传参时 ...
- springMVC常用传参总结
本文介绍了springMVC常用的传参方式和一些注意的事项,页面表单主要以ajax的形式提交. 本帅是个菜鸡,水平有限,若有什么讲得不对或有补充的地方欢迎各位提意见. 一.传递String类型 1 ...
- (三)Mybatis类型转换器,接口传参类型,一对一,一对多查询resultMap配置
Mybatis类型转换器 首先明白什么时候用到它,当数据库的字段类型和java字段类型无法默认匹配时候进行转换,比如现在数据库类型是INTEGER,而java当中类型是Boolean,true表示1, ...
- mybatis-plus 自定义SQL,XML形式,传参的几种方式
mybatis-plus 自定义SQL,XML形式,传参的几种方式 前提说明 所涉及文件 传参类型说明 1.Java代码中使用QueryWrapper动态拼装SQL 2.简单类型参数(如String, ...
- SpringMVC前后端分离交互传参详细教程
温故而知新,本文为一时兴起写出,如有错误还请指正 本文后台基于SpringBoot2.5.6编写,前端基于Vue2 + axios和微信小程序JS版分别编写进行联调测试,用于理解前后端分离式开发的交互 ...
- 浅谈对java中传参问题的理解
之前用的c/c++比较多,在c/c++中对于传参类型,无外乎就是传值.传引用.传指针这几种.但在java中,由于没有指针类型,其传参的方式也发生了相应的变化.在网上找了找,按我之前的理解,java中传 ...
- Java的对象传参问题
在c/c++中对于传参类型,无外乎就是传值.传引用.传指针这几种.但在java中,由于没有指针类型,其传参的方式也发生了相应的变化.之前有搜过相关的知识点一直理解的是:Java的传参方式中主要有两种: ...
随机推荐
- MySQL内存表的特性与使用介绍
国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...
- C++ “读取位置 0x****** 时发生访问冲突”的可能原因
转自:http://shansun123.iteye.com/blog/680066 这种错误的意思一般是指访问了不属于自己的内存空间,出现这种错误有几种原因: 1.给一个数组分配了比较小的内存空间, ...
- 文件和目录之utime函数
一个文件的访问和修改时间可以用utime函数更改. #include <utime.h> int utime( const char *pathname, const struct uti ...
- codereview介绍
1. 定义: Code review is systematic examination (often known as peer review) of computer source code. I ...
- sass+require实现侧边栏
一.效果图(如下)及使用的技术 实现用sass实现页面中右侧固定侧边栏的样式,用require.js实现返回顶部的功能 二.sass 具体的sass的介绍就不多说了,大家可以参考sass官网介绍,下面 ...
- Android_AutoCompleteTextView,MultiAutoCompleteTextView
XML布局文件 <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:an ...
- 在oc代码中使用swift第三方框架
swift现在使用越来越多了,一些主流的框架或者效果比较好的demo都陆陆续续使用swift写了.所以,要学会如何在oc的项目中调用swift. 这里主要借助一个桥梁文件,这个桥梁文件一般在你导入sw ...
- 修改UILabel的行间距
在iOS开发中 有时候为了调整一些UI效果 我们需要调整UILabel之间的行间距: contentLabel.text:label上显示的文字内容; 5:label行间距; contentLab ...
- Lucene 搜索功能
搜索过程 图解: 主要 API: IndexSearcher: //所有搜索都通过 IndexSearcher 进行,他们将调用该类中重载的 search() 方法 Query: ...
- Spring MVC中基于注解的 Controller
终于来到了基于注解的 Spring MVC 了.之前我们所讲到的 handler,需要根据 url 并通过 HandlerMapping 来映射出相应的 handler 并调用相应的方法以响 ...