C++: unordered_map 花式插入key-value的5种方式

前言

无意中发现std::unordered_map、std::map等插入key-value对在C++17后竟有了 insert()operator[]emplace()try_emplace()insert_or_assign() 等超过5种方法,我们可以根据实际场景和对效率的要求,去选择不同的方法。在此不得不夸一夸C++的灵(fù)活(zá)性,不管怎么说,一点无用的知识又增加了。此外发现,Effective STL这本书中对insert()方法的介绍有些过时了。

下文中,使用一个测试类作为关联容器中的 mapped_type 来探究通过不同方法对map进行值插入的开销。

测试类定义

测试类 MyClass 定义如下, id 用于标识不同实例,定义了构造函数、拷贝构造函数、赋值运算符函数,构造函数设置 explicit 不允许隐式构造, 未定义移动构造函数和移动赋值函数。

class MyClass {
private:
int id = 0;
public:
MyClass() {
std::cout << "Default Constructor called " << id << "\n";
}
explicit MyClass(int i):id(i) {
std::cout << "Constructor called " << id << "\n";
} MyClass(const MyClass& my_class) {
id = my_class.id;
std::cout<< "Copy Constructor called "<< id << "\n";
} MyClass& operator=(const MyClass& my_class) {
id = my_class.id;
std::cout<< "Operator= called "<< id <<"\n";
return *this;
} ~MyClass() {
std::cout<<"Destructor called "<<id<<"\n";
}
};

初始化一个 unordered_map 和一些自定义类 MyClass的对象:

std::unordered_map<std::string, MyClass> myMap;
MyClass m1(1), m2(2),m3(3),m4(4),m5(5),m6(6),m7(7),m8(8),m9(9);

测试对比

将插入元素分为 add(key不存在)和 update (key已存在)两种情况进行讨论,基于 myMap 依次运行以下代码,对比相关函数调用开销。

1. add ,key不存在

依次按以下代码顺序执行插入 key-value对:

  • insert() :
myMap.insert({"one", m1});

创建临时的 key-value node 以及将其拷贝进 myMap 容器,二者都会调用MyClass的拷贝构造函数(本应移动MyClass,但未定义移动操作只能拷贝)。调用该方法后输出如下:

Copy Constructor called 1 // Make tmp std::pair
Copy Constructor called 1 // Copy pair to container
Destructor called 1 // tmp MyClass de
  • operator[] :
 myMap["two"] = m2;

该方法要求 mapped_type是可默认构造的, 当key不存在时,在myMap 中先分配了一个 {key, MyClass()} node的空间,该运算符返回该 MyClass() 的引用,再用 MyClass(2) 进行赋值,此过程调用默认构造函数和赋值运算符函数。调用该方法后输出如下:

Default Constructor called 0  // Call Default Constructor
Operator= called 2 // Call Operator=()
  • emplace() :
myMap.emplace("three", m3);

直接传入key-value,在容器中原地构造 std::pair ,省去了相关函数调用开销。

Copy Constructor called 3      // Copy MyClass(3) to myMap

总结:当对效率要求较高,key不存在时,应优先使用 emplace() 插入key-value,避免临时变量带来的开销。

2.update,key存在

  • operator[] :

当Key存在时,value会被替换为新值,

 myMap["one"] = m4;

以上代码仅调用赋值运算符函数。

Operator= called 4
  • insert()emplace()

这两种方法,当Key存在时,value不会被替换为新值。但临时值会被创建出来。

myMap.insert(std::make_pair("one", m4));

insert() 而言,创建临时key-value node以及拷贝进容器的操作都会执行。

Copy Constructor called 5
Copy Constructor called 5
Destructor called 5
Destructor called 5

emplace() 而言,继续做以下插入操作,

myMap.emplace("three", m6);

“three”对应的value仍为 m3 ,但pair的临时变量仍然会被创建,之后便销毁:

Copy Constructor called 6
Destructor called 6
  • C++17 try_emplace() :

如果key已经存在,不会创建key-value node。否则,将会将其插入到map中,

myMap.try_emplace("three", m7);

以上代码输出结果如下,未创建pair。

Constructor called 7
Destructor called 7
  • C++17 insert_or_assign()

当key存在时,将对应value值进行更新插入key-value对,

myMap.insert_or_assign("three", m8);

运行以上代码后,”three”对应value为 m8 , 输出如下, 仅调用了赋值运算符函数,这与 operator[]是一样的。

Operator= called 8

当key不存在时,插入key-value对,

myMap.insert_or_assign("four", m9);

运行后输出如下,仅调用拷贝构造函数,可见,该方法也支持原地构造。与 operator[] 不同的是,该方法不需要 mapped_type 支持默认构造函数。

Copy Constructor called 9
  • operator[] vs insert_or_assign() :
要求Value可默认构造 返回值
operator[] true value
insert_or_assign() false pair<iterator, bool>

insert_or_assign() 的返回值为 std::pair<iterator, bool> ,其中 iterator 指向插入或更新的元素, bool 变量的含义为:如果发生插入,值为 true ;如果发生替换,值为 false

总之,当key存在时,如果需要替换value值,应使用operator[] ;需要更丰富的返回信息时,可考虑insert_or_assign() 。如果不需要替换value值,为避免临时node创建,可使用 try_emplace()

测试程序地址https://godbolt.org/z/M3KTPhvoY

总结

以上提到的5种方法之间的差异对比可参考下图

各方法对比如下:

C++版本 是否覆盖value 构造node前事先查找
insert() C++03 false false
operator[] C++03 true \
emplace() C++11 false false
try_emplace() C++17 false true
insert_or_assign() C++17 true \

最后总结,当对效率要求较高:

  • 当key不存在时,应优先使用 emplace() 插入key-value,避免创建临时变量带来的开销。

  • 当key存在时,如果需要替换value值,应使用operator[] ;如果需要更丰富的返回信息时,可考虑insert_or_assign()

  • 当key存在时,现代C++的 insert()方法已经不能更新值了,Effective STL书中的介绍已经过时。

  • 如果不需要替换value值,为避免临时node创建,可使用 try_emplace()

References

  1. https://en.cppreference.com/w/cpp/container/unordered_map
  2. https://www.fluentcpp.com/2018/12/11/overview-of-std-map-insertion-emplacement-methods-in-cpp17/
  3. https://en.cppreference.com/w/cpp/container/map/insert_or_assign

你好,我是七昂,致力于分享C++、计算机底层、机器学习等系列知识。希望我们能一起探索程序员修炼之道。如果我的创作内容对您有帮助,请点赞关注。如果有问题,欢迎随时与我交流。感谢你的阅读。

公众号: 七昂的技术之旅

C++: 如何高效地往unordered_map中插入key-value的更多相关文章

  1. js向一个数组中插入元素的几个方法-性能比较

    向一个数组中插入元素是平时很常见的一件事情.你可以使用push在数组尾部插入元素,可以用unshift在数组头部插入元素,也可以用splice在数组中间插入元素. 但是这些已知的方法,并不意味着没有更 ...

  2. 如何通过Java代码在PDF中插入、替换或删除图像?

    图文并茂的内容往往让人看起来更加舒服,如果只是文字内容的累加,往往会使读者产生视觉疲劳.搭配精美的文章配图则会使文章内容更加丰富,增加文章可读性的同时,也能提升用户体验.但由于PDF文档安全性较高,不 ...

  3. 如何在latex 中插入EPS格式图片

    如何在latex 中插入EPS格式图片 第一步:生成.eps格式的图片 1.利用visio画图,另存为pdf格式的图片 利用Adobe Acrobat裁边,使图片大小合适 另存为.eps格式,如下图所 ...

  4. Latex中插入C语言代码

    Latex是一个文本排版的语言,能排版出各种我们想要的效果.而且用代码排版的优点是易于修改板式,因此在文本内容的排版时,Latex应用十分广泛. 当我们需要在Latex中插入代码时,就需要用到 \us ...

  5. 【转载】在HTML中插入swf文件(转)

    在HTML中插入swf文件(转) 在网页里面插入swf,再平常不过了,一般会想到如下代码: Html代码 <object classid="clsid:D27CDB6E-AE6D-11 ...

  6. 在MySQL向表中插入中文时,出现:incorrect string value 错误

    在MySQL向表中插入中文时,出现:incorrect string value 错误,是由于字符集不支持中文.解决办法是将字符集改为GBK,或UTF-8.      一.修改数据库的默认字符集   ...

  7. SQL语句 在一个表中插入新字段

    SQL语句 在一个表中插入新字段: alter table 表名 add 字段名 字段类型 例: alter table OpenCourses add Audio varchar(50)alter ...

  8. Markdown中插入数学公式的方法

    Markdown中插入数学公式的方法 文章来源:http://blog.csdn.net/xiahouzuoxin/article/details/26478179 自从使用Markdown以来,就开 ...

  9. java POI实现向Excel中插入图片

          做Web开发免不了要与Excel打交道.今天老大给我一个任务-导出Excel.开始想的还是蛮简单的,无非就是查找,构建Excel,response下载即可.但是有一点不同,就是要加入图片, ...

  10. 使用C#向ACCESS中插入数据

    使用C#向ACCESS中插入数据   1.创建并打开一个OleDbConnection对象 string strConn = " Provider = Microsoft.Jet.OLEDB ...

随机推荐

  1. Aspose Excel 单元格合并后边框显示不全

    /// <summary> /// 解决合并后的单元格没有边框,设置合并单元格格式,让合并过的单元格中每一个单元格上都添加上加边框的样式 /// </summary> /// ...

  2. 重磅来袭!MoneyPrinterPlus一键发布短视频到视频号,抖音,快手,小红书上线了

    MoneyPrinterPlus开源有一段时间了,已经实现了批量短视频混剪,一键生成短视频等功能. 有些小伙伴说了,我批量生成的短视频能不能一键上传到视频号,抖音,快手,小红书这些视频平台呢?答案是必 ...

  3. Vue 3 后端错误消息处理范例

    1. 错误消息格式 前后端消息传递时,我们可以通过 json 的 errors 字段传递错误信息,一个比较好的格式范例为: { errors: { global: ["网络错误"] ...

  4. WebGL管网展示(及TubeGeometry优化)

    前言 管路展示在三维场景中很常见.比如地下管网,建筑里面的水果,暖通管道等等的展示. 建立管路的方式主要两种: 通过3DMax C4D Blender等建模工具进行建模. 通过路径数据,程序生成三维管 ...

  5. tp5.1--数据库事务操作

    https://blog.csdn.net/qq_42176520/article/details/88708395 使用事务处理的话,需要数据库引擎支持事务处理.比如 MySQL 的 MyISAM  ...

  6. [oeasy]python0096_游戏娱乐行业_雅达利_米洛华_四人赛马_影视结合游戏

    游戏娱乐行业 回忆上次内容 游戏机行业从无到有 雅达利 公司 一枝独秀 并且带领 行业 发展起来 雅达利公司 优秀员工 乔布斯 在 朋友 帮助下完成了<pong> Jobs 黑了 Woz ...

  7. TIER 1: Sequel

    TIER 1: Sequel MySQL MySQL 是一种流行的开源关系型数据库管理系统(RDBMS),它被广泛用于存储和管理大量的结构化数据.MySQL 是由瑞典公司 MySQL AB 开发的,后 ...

  8. 【转载】Win10系统, administrator账户被微软账户强行绑定,怎么破?

    首先 声明:这是转载,我只是做一个记录,以下内容可解决问题(本人已尝试并已解决),当然也可以去转载出处查看大佬的原回答: Win10系统, administrator账户被微软账户强行绑定,怎么破? ...

  9. sharding-jdbc 兼容 MybatisPlus的动态数据源

    背景:之前的项目做读写分离的时候用的 MybatisPlus的动态数据做的,很多地方使用的@DS直接指定的读库或者写库实现的业务:随着表数据量越来越大,现在打算把比较大的表进行水平拆分,准备使用 Sh ...

  10. Postman汉化成中文版

    postman安装默认是英文版,为使用方便使用汉化包转成中文版 1.查看本地安装的postman版本:Settings->About 2.根据postman的版本下载对应的汉化包,汉化包网址 3 ...