C++: 如何高效地往unordered_map中插入key-value
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[]
vsinsert_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
- https://en.cppreference.com/w/cpp/container/unordered_map
- https://www.fluentcpp.com/2018/12/11/overview-of-std-map-insertion-emplacement-methods-in-cpp17/
- https://en.cppreference.com/w/cpp/container/map/insert_or_assign
你好,我是七昂,致力于分享C++、计算机底层、机器学习等系列知识。希望我们能一起探索程序员修炼之道。如果我的创作内容对您有帮助,请点赞关注。如果有问题,欢迎随时与我交流。感谢你的阅读。
公众号: 七昂的技术之旅
C++: 如何高效地往unordered_map中插入key-value的更多相关文章
- js向一个数组中插入元素的几个方法-性能比较
向一个数组中插入元素是平时很常见的一件事情.你可以使用push在数组尾部插入元素,可以用unshift在数组头部插入元素,也可以用splice在数组中间插入元素. 但是这些已知的方法,并不意味着没有更 ...
- 如何通过Java代码在PDF中插入、替换或删除图像?
图文并茂的内容往往让人看起来更加舒服,如果只是文字内容的累加,往往会使读者产生视觉疲劳.搭配精美的文章配图则会使文章内容更加丰富,增加文章可读性的同时,也能提升用户体验.但由于PDF文档安全性较高,不 ...
- 如何在latex 中插入EPS格式图片
如何在latex 中插入EPS格式图片 第一步:生成.eps格式的图片 1.利用visio画图,另存为pdf格式的图片 利用Adobe Acrobat裁边,使图片大小合适 另存为.eps格式,如下图所 ...
- Latex中插入C语言代码
Latex是一个文本排版的语言,能排版出各种我们想要的效果.而且用代码排版的优点是易于修改板式,因此在文本内容的排版时,Latex应用十分广泛. 当我们需要在Latex中插入代码时,就需要用到 \us ...
- 【转载】在HTML中插入swf文件(转)
在HTML中插入swf文件(转) 在网页里面插入swf,再平常不过了,一般会想到如下代码: Html代码 <object classid="clsid:D27CDB6E-AE6D-11 ...
- 在MySQL向表中插入中文时,出现:incorrect string value 错误
在MySQL向表中插入中文时,出现:incorrect string value 错误,是由于字符集不支持中文.解决办法是将字符集改为GBK,或UTF-8. 一.修改数据库的默认字符集 ...
- SQL语句 在一个表中插入新字段
SQL语句 在一个表中插入新字段: alter table 表名 add 字段名 字段类型 例: alter table OpenCourses add Audio varchar(50)alter ...
- Markdown中插入数学公式的方法
Markdown中插入数学公式的方法 文章来源:http://blog.csdn.net/xiahouzuoxin/article/details/26478179 自从使用Markdown以来,就开 ...
- java POI实现向Excel中插入图片
做Web开发免不了要与Excel打交道.今天老大给我一个任务-导出Excel.开始想的还是蛮简单的,无非就是查找,构建Excel,response下载即可.但是有一点不同,就是要加入图片, ...
- 使用C#向ACCESS中插入数据
使用C#向ACCESS中插入数据 1.创建并打开一个OleDbConnection对象 string strConn = " Provider = Microsoft.Jet.OLEDB ...
随机推荐
- 你有对 Vue 项目进行哪些优化?
(1)代码层面的优化 v-if 和 v-show 区分使用场景 computed 和 watch 区分使用场景 v-for 遍历必须为 item 添加 key,且避免同时使用 v-if 图片资源懒加载 ...
- 小产品,快变现,Solo社区共建者 James 专访
采访人:徐小夕. 本次受邀采访的嘉宾是Solo社区计划负责人&Solo社区联合创建者 James Pan(老潘). 专访内容 1. Solo社区创建的背景 随着国内软件市场内卷加剧,加上大环境 ...
- [oeasy]python0129_unicode_中文字符序号_十三道大辙_字符编码解码_eval_火星文
unicode 中文字符分类 回忆上次内容 字符集 从博多码 到 ascii 再到 iso-8859 系列 各自割据 如何把世界上各种字符统进行编码 unicode顺势而生不断进化 不过字符总量超 ...
- oeasy教您玩转vim - 15 - # 行内查找
行头行尾 回忆上节课内容 上次学了直接跳到开头和结尾 最重要的就是 ^.$ ^ 到开头 $ 到结尾 I 相当于^i A 相当于$a 查找帮助 还有什么呢? 还是继续在 motion 里面 ^ .$ 之 ...
- AT_arc154_b 题解
洛谷链接&Atcoder 链接 本篇题解为此题较简单做法及较少码量,并且码风优良,请放心阅读. 题目简述 给定两个长度为 \(n\) 的字符串 \(S,T\),定义一次操作可取出 \(S\) ...
- 关键点检测(1)——标注关键点检测数据(labelme和CVAT)
关键点检测,作为计算机视觉领域的重要分支,广泛应用于人体姿态估计.面部表情识别.手部动作分析等多个场景.其核心在于从图像中准确检测并定位特定的关键点位置.然而,高效的模型训练离不开大量高质量的标注数据 ...
- 推荐几款.NET开源且功能强大的实用工具,助你提高工作开发效率!
前言 俗话说得好"工欲善其事,必先利其器",今天大姚给大家推荐8款.NET开源且功能强大的实用工具,助你提高工作开发效率! DevToys 一款基于C#开源(MIT License ...
- C语言中的断言函数assert
简介 assert 是 C 语言中的一个宏,用于在程序运行时进行条件检查,主要用于调试目的.它在 <assert.h> 头文件中定义,用于验证程序中的假设条件是否成立,如果不成立,程序将打 ...
- 使用MySQL实现分布式锁
分布式锁开发中经常使用,在项目多节点部署或者微服务项目中,JAVA提供的线程锁已经不能满足安全的需求,需要使用全局的分布式锁来保证安全:分布式锁的实现的方式有很多种,最常见的有zookeeper,Re ...
- Jmeter参数化1-随机数设置
背景:当新增接口的某个字段是唯一性,每次调用该新增接口都会需要单独传入这个字段,麻烦且繁琐. 解决:jmeter设置随机数参数,然后接口调用该参数就达到了自动性不再需要人工传入不同的值.方便调用接口, ...