std::map<Key,T,Compare,Allocator>::operator[]
来自cppreference.com
T& operator[]( const Key& key ); |
(1) | |
T& operator[]( Key&& key ); |
(2) | (C++11 起) |
template< class K > T& operator[]( K&& x ); |
(3) | (C++26 起) |
返回到映射到分别等于 key 或 x 的键的值的引用,这种键不存在的情况下就会进行插入。
1) 在键不存在的情况下插入 value_type(key, T())。
|
(C++11 前) | ||||||
1) 在键不存在的情况下插入从 std::piecewise_construct, std::forward_as_tuple(key), std::tuple<>() 原位构造的
value_type 对象。 等价于 return this->try_emplace(key).first->second;。 (C++17 起)使用默认分配器时,这导致从 key 复制构造键,并值初始化被映射值。
2) 在键不存在的情况下插入从 std::piecewise_construct, std::forward_as_tuple(std::move(key)), std::tuple<>() 原位构造的
value_type 对象。 等价于 return this->try_emplace(std::move(key)).first->second; 。 (C++17 起) 使用默认分配器时,这导致从 key 移动构造键,并值初始化被映射值。
|
(C++11 起) |
3) 如果没有与 x 的值进行透明比较相等的键,则插入一个就地构造的
value_type
对象。 等价于 return this->try_emplace(std::forward<K>(x)).first->second;。
此重载仅在 此重载只有在限定标识 Compare::is_transparent 合法并指代类型时才会参与重载决议。它允许调用此函数时无需构造
Key
的实例。时参与重载决议。它允许调用该函数时无需构建 Key
的实例。没有迭代器或引用会失效。
参数
key | - | 要寻找的元素键 |
x | - | 任何可以和键进行透明比较的类型的值 |
返回值
1,2) 不存在拥有键 key 的元素时返回到新元素被映射值的引用。否则返回指代其键等价于 key 的既存元素的被映射值的引用。
3) 不存在拥有与 x 值比较等价的键的元素时返回到新元素被映射值的引用。否则返回指代其键与 x 比较等价的既存元素的被映射值的引用。
异常
如果任何操作抛出异常,那么插入无效果。
复杂度
与容器大小成对数。
注解
出版的 C++11 和 C++14 标准中,指定此函数要求 mapped_type
为可默认插入 (DefaultInsertable) 且 key_type
为可复制插入 (CopyInsertable) 或可移动插入 (MoveInsertable) 到 *this。此规定有缺陷并为 LWG 问题 2469 所修复,而上面的描述合并了该问题的解决方案。
然而,已知一个实现(libc++)通过两个分离的分配器 construct()
调用来构造 key_type
和 mapped_type
对象,可认为如发布时的标准所要求,而非原位构造 value_type
对象。
operator[] 非 const,因为它会在键不存在时插入键。如果此行为非所欲或容器为 const,那么可以使用 at
。
|
(C++17 起) |
功能特性测试宏 | 值 | 标准 | 功能特性 |
---|---|---|---|
__cpp_lib_associative_heterogeneous_insertion |
202311L | (C++26) | 剩余的有序和无序关联容器的成员函数的异质重载。(3) |
示例
运行此代码
#include <iostream> #include <string> #include <map> void println(auto const comment, auto const& map) { std::cout << comment << "{"; for (const auto &pair : map) std::cout << "{" << pair.first << ": " << pair.second << "}"; std::cout << "}\n"; } int main() { std::map<char, int> letter_counts{{'a', 27}, {'b', 3}, {'c', 1}}; println("letter_counts 起初包含:", letter_counts); letter_counts['b'] = 42; // 更新既存值 letter_counts['x'] = 9; // 插入新值 println("修改后它包含:", letter_counts); // 统计每个单词的出现次数 // (首次调用 operator[] 会初始化计数为零) std::map<std::string, int> word_map; for (const auto& w : {"this", "sentence", "is", "not", "a", "sentence", "this", "sentence", "is", "a", "hoax"}) ++word_map[w]; word_map["that"]; // 插入对 {"that", 0} for (const auto& [word, count] : word_map) std::cout << "单词 '" << word << "' 出现 " << count << "次\n"; }
输出:
letter_counts 初始状态下包含:{{a: 27}{b: 3}{c: 1}} 修改后它包含:{{a: 27}{b: 42}{c: 1}{x: 9}} 单词 'a' 出现 2 次 单词 'hoax' 出现 1 次 单词 'is' 出现 2 次 单词 'not' 出现 1 次 单词 'sentence' 出现 3 次 单词 'that' 出现 0 次 单词 'this' 出现 2 次
缺陷报告
下列更改行为的缺陷报告追溯地应用于以前出版的 C++ 标准。
缺陷报告 | 应用于 | 出版时的行为 | 正确行为 |
---|---|---|---|
LWG 334 | C++98 | 重载 (1) 的效果只是简单地返回 (*((insert(std::make_pair(x, T()))).first)).second |
自己提供效果描述 |
参阅
带越界检查访问指定的元素 (公开成员函数) | |
(C++17) |
插入元素,或若键已存在则赋值给当前元素 (公开成员函数) |
(C++17) |
若键不存在则原位插入,若键存在则不做任何事 (公开成员函数) |