在C++20的标准范围库中,std::views::values是专门用来从键值对类型的范围中提取所有值元素,生成对应值视图的工具,无需额外拷贝数据就能直接操作原容器中的值部分,配合管道操作符可以大幅简化对map类容器的处理逻辑。

std::views::values基本用法
std::views::values属于标准范围适配器,接收的参数需要是包含键值对的范围,比如std::map、std::unordered_map、或者std::vector<std::pair<K, V>>这类存储键值对元素的容器。提取后的视图元素是原容器中每个键值对的second值,并且视图是惰性求值的,不会立即生成新的容器,只有在遍历或者消费视图时才会实际取值。
下面是一个最基础的使用示例,展示如何提取map中的所有值:
#include <iostream>
#include <map>
#include <ranges>
#include <vector>
int main() {
std::map<int, std::string> student_map = {
{1, "张三"},
{2, "李四"},
{3, "王五"}
};
// 提取map的值视图
auto values_view = std::views::values(student_map);
// 遍历值视图输出所有名字
for (const auto& name : values_view) {
std::cout << name << std::endl;
}
return 0;
}
上述代码中,values_view是一个视图对象,遍历它的时候会依次取出student_map中每个元素的second值,也就是三个学生的名字,输出结果分别是张三、李四、王五。
结合C++20管道操作的使用技巧
C++20的管道操作符|可以将多个范围适配器串联起来,std::views::values作为适配器可以直接放在管道链中,先提取值再做后续处理,比如过滤、转换、取前N个元素等,代码逻辑会非常清晰。
技巧1:提取后过滤符合条件的值
假设我们需要从map中提取值,并且只保留长度大于2的字符串,就可以把std::views::values和std::views::filter结合起来使用:
#include <iostream>
#include <map>
#include <ranges>
#include <string>
int main() {
std::map<int, std::string> data_map = {
{1, "a"},
{2, "bb"},
{3, "ccc"},
{4, "dddd"}
};
// 管道操作:先提取值,再过滤长度大于2的字符串
auto filtered_view = data_map
| std::views::values
| std::views::filter([](const std::string& s) {
return s.size() > 2;
});
for (const auto& s : filtered_view) {
std::cout << s << std::endl;
}
return 0;
}
这个例子中,首先通过data_map | std::views::values得到所有值的视图,再通过管道传给filter适配器,筛选出长度大于2的字符串,最终遍历输出ccc和dddd两个结果。
技巧2:提取值后做转换处理
如果需要把提取出来的数值做统一转换,比如给所有值加1,可以结合std::views::transform使用:
#include <iostream>
#include <map>
#include <ranges>
int main() {
std::map<int, int> num_map = {
{1, 10},
{2, 20},
{3, 30}
};
// 管道操作:提取值后每个值加1
auto transformed_view = num_map
| std::views::values
| std::views::transform([](int val) {
return val + 1;
});
for (int val : transformed_view) {
std::cout << val << " ";
}
// 输出结果:11 21 31
return 0;
}
技巧3:限制提取值的数量
如果只需要取前N个值,可以结合std::views::take使用:
#include <iostream>
#include <map>
#include <ranges>
int main() {
std::map<int, int> num_map = {
{1, 10},
{2, 20},
{3, 30},
{4, 40}
};
// 取前2个值
auto take_view = num_map
| std::views::values
| std::views::take(2);
for (int val : take_view) {
std::cout << val << " ";
}
// 输出结果:10 20
return 0;
}
使用注意事项
- std::views::values生成的视图是只读的,不能修改原容器中的值,如果需要修改原容器的值,需要使用std::views::values的结果配合引用,但是要注意原容器的生命周期,视图不会延长原容器的生命周期,原容器销毁后视图会变成悬垂状态。
- std::views::values只能处理键值对类型的范围,如果传入的容器元素不是pair或者类似键值对的结构,编译会报错。
- 视图是惰性求值的,多次遍历同一个视图的时候,每次都会重新从原容器取值,所以如果原容器在两次遍历之间被修改,视图的内容也会对应变化。
适用场景
当需要单独处理map、unordered_map或者其他键值对容器中的值部分,并且不需要修改原容器结构,同时需要做过滤、转换等后续操作时,使用std::views::values配合管道操作会比传统的for循环加条件判断的代码更简洁,可读性更高,而且不需要额外的拷贝开销,适合性能要求较高的场景。
std::views::valuesmap值视图C++20管道操作C++20修改时间:2026-06-16 18:12:36