您的位置:首页 >C++ std::views::keys与values _ 快速获取Map的键或值列表【详解】
发布于2026-05-02 阅读(0)
扫一扫,手机访问

简单来说,std::views::keys 不能直接用于 std::map,主要是因为早期编译器(比如 GCC 版本低于12、Clang 低于15、MSVC 低于17.3)对带有 const 键的 map 迭代器支持不够完善。要顺利使用,你得确保编译器版本达标、包含了必要的头文件( 和 ),并且要小心别在移动 map 后使用视图,还得时刻留意视图的生命周期问题。
这事儿得从底层机制说起。std::views::keys 要求它所处理的元素,必须是那种能“拆解”成键值对的结构。换句话说,元素类型得支持像 std::get<0>(e) 这样的操作,或者拥有 .first 和 .second 这样的成员。
那么问题来了:std::map 的迭代器解引用后,得到的是 std::pair。注意看,这里的键是 const Key。从语法上讲,这完全符合要求。真正的“坑”往往出现在视图适配器与 const 限定符的交互上。
具体来说,如果你用的是 C++20 早期的编译器实现(比如 GCC 11 或者 MSVC 2022 的 17.0 版本之前),std::views::keys 可能还没能完全处理好这种带有 const 键的 map 迭代器。直到 GCC 12 及以上、Clang 15 及以上、MSVC 17.3 及以上版本,这个支持才算是稳定下来。
所以,想避开这个坑,记住下面几点:
#include 和 #include 。views::keys——移动后的 map 迭代器行为是未定义的。答案是引用,不是副本。std::views::values 返回的视图元素类型,就是原容器中 value 的引用类型(T&)。当然,如果原容器本身是 const 的,那你得到的自然就是 const T&。
这里有个常见的理解误区:当你写下 for (auto v : std::views::values(m)) 这样的循环时,如果 m 是一个 std::unordered_map,那么 v 的类型其实是 int&。为了避免在循环中不小心修改了原值,更安全的做法是显式声明为 const auto& v。
几个关键点需要牢记:
std::views::values(m)[0] = 42; 是合法的,并且会生效(前提是 m 不是 const 对象)。m 是一个临时对象(比如函数返回值),那么 std::views::values(m) 会绑定到这个临时对象的生命周期上。但视图本身并不会延长临时对象的生命,之后使用这个视图就会导致悬垂引用,这是非常危险的。std::views::values 本身不支持随机访问(即 operator[]),除非底层容器的迭代器是随机访问迭代器。所以,对于 std::vector 可以,但对于 std::map 就不行。立即学习“C++免费学习笔记(深入)”;
其实很简单,直接用 std::vector 的 range 构造函数就可以了。这个构造函数是单次遍历的——它只会调用一次 begin() 和 end(),然后在内部按需推进迭代器来填充元素。
std::mapm = {{1,"a"}, {2,"b"}, {3,"c"}}; std::vector keys_vec(std::views::keys(m)); // ✅ 单次遍历
要避免的错误写法是:std::vector。这实际上创建了两个独立的视图对象,可能会引发重复计算,尤其是在处理自定义范围(range)时。
std::vector,然后再使用 std::sort 或 std::unique。直接对 std::views::keys(m) 使用 std::views::sort 是不合法的,因为 keys 视图本身不可排序。std::map 本身按键排序,所以 std::views::keys(m) 得到的键序列天然就是有序的。而对于 std::unordered_map,顺序是不确定的,并且不能保证两次遍历的顺序一致。for (...) { auto ks = std::views::keys(m); /* ... */ } 这样的写法,最好将视图的构造提取到循环外部。基本上没有额外的特殊要求。只要这个 key 类型本身能被 std::map 存储(即满足 std::is_move_constructible_v 等基本约束),就完全没问题。
但这里有个非常重要的细节:视图并不拷贝 key,它只是提供对原有 key 的引用。所以,即使 key 是一个大对象(比如很长的 std::string 或者自定义的结构体),使用 std::views::keys 依然非常轻量,不会引发字符串复制或深拷贝。
真正容易被忽略的,是生命周期绑定问题。当你写下 auto keys_view = std::views::keys(m); 时,这个 keys_view 视图隐式地持有着对原始 map m 的引用。一旦 m 被销毁(比如离开了作用域)或者被重新赋值,keys_view 就变成了一个悬垂视图(dangling view)。此时使用它会导致未定义行为,而且编译器通常不会报错,这给调试带来了很大困难。
std::views::keys 结果返回给函数调用方。static_assert(std::is_reference_v) 来静态断言确认。上一篇:如何在 Laravel Blade 中实现过滤后仍保持正确的奇偶交替样式
下一篇:C#怎么操作Timer定时器控件 C#WinForms Timer和Threading.Timer的区别和使用场景【控件】
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
正版软件
正版软件
正版软件
正版软件
正版软件
1
2
3
7
9