C++中的std::ranges::pipe是什么?如何像管道一样组合算法

来源:个人站长作者:桃乃木香奈头衔:网络博主
导读:本期聚焦于小伙伴创作的《C++中的std::ranges::pipe是什么?如何像管道一样组合算法》,敬请观看详情,探索知识的价值。以下视频、文章将为您系统阐述其核心内容与价值。如果您觉得《C++中的std::ranges::pipe是什么?如何像管道一样组合算法》有用,将其分享出去将是对创作者最好的鼓励。

std::ranges::pipe是C++20标准中ranges库提供的一个工具,它的核心作用是让多个针对范围(range)的操作可以像管道一样依次串联起来,形成一个连贯的处理流程,避免传统写法中多层嵌套或者临时变量的冗余问题。

std::ranges::pipe的基本概念

在C++20之前,如果我们需要对一组数据进行筛选、转换、排序等一系列操作,通常需要先调用一个算法得到临时结果,再把临时结果传给下一个算法,代码会显得非常繁琐。而std::ranges::pipe提供了一种链式调用的方式,让这些操作可以像管道传输数据一样依次执行。

它本身并不是一个具体的函数或者类,而是一种可以被重载的运算符相关的机制,只要自定义的类型或者标准库中的范围适配器支持管道运算符|,就可以和std::ranges::pipe配合实现组合调用。

标准库中的范围适配器管道用法

C++标准库已经为很多常用的范围操作实现了适配器,这些适配器都支持通过管道运算符|来组合,下面是几个常见的示例。

示例1:筛选和转换的组合

我们有一个整数向量,需要筛选出所有偶数,然后将这些偶数乘以2,最后输出结果。

#include <iostream>
#include <vector>
#include <ranges>

int main() {
    std::vector<int> nums = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
    // 使用管道组合筛选和转换操作
    auto result = nums 
        | std::views::filter([](int n) { return n % 2 == 0; })  // 筛选偶数
        | std::views::transform([](int n) { return n * 2; });  // 每个偶数乘2
    // 输出结果
    for (int n : result) {
        std::cout << n << " ";
    }
    // 输出内容:4 8 12 16 20
    return 0;
}

示例2:更多操作的管道组合

我们可以在管道中加入更多操作,比如先筛选,再转换,最后取前3个元素。

#include <iostream>
#include <vector>
#include <ranges>

int main() {
    std::vector<int> nums = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
    auto result = nums
        | std::views::filter([](int n) { return n > 3; })       // 筛选大于3的数
        | std::views::transform([](int n) { return n * 10; })   // 每个数乘10
        | std::views::take(3);                                  // 取前3个元素
    for (int n : result) {
        std::cout << n << " ";
    }
    // 输出内容:40 50 60
    return 0;
}

自定义类型支持std::ranges::pipe

如果我们自己实现了一个范围适配器,也可以通过重载管道运算符让它能够参与管道组合。要实现这一点,需要满足一定的条件:自定义的类型需要是一个可调用对象,并且重载operator|运算符,让它能够和范围或者其他适配器正确组合。

下面是一个简单的自定义适配器的示例,这个适配器的作用是将范围中的每个元素加上一个固定的值。

#include <iostream>
#include <vector>
#include <ranges>
#include <concepts>

// 自定义适配器,给每个元素加固定值
struct add_value {
    int val;
    // 调用运算符,处理范围
    auto operator()(std::ranges::input_range auto&& r) const {
        return r | std::views::transform([this](int n) { return n + val; });
    }
    // 重载管道运算符,支持范围直接通过管道调用适配器
    friend auto operator|(std::ranges::input_range auto&& r, const add_value& adder) {
        return adder(std::forward<decltype(r)>(r));
    }
};

int main() {
    std::vector<int> nums = {1, 2, 3, 4};
    // 使用自定义适配器参与管道组合
    auto result = nums 
        | std::views::filter([](int n) { return n % 2 == 1; })  // 筛选奇数
        | add_value{5};                                         // 每个奇数加5
    for (int n : result) {
        std::cout << n << " ";
    }
    // 输出内容:6 8
    return 0;
}

std::ranges::pipe的注意事项

  • 管道组合的操作都是惰性求值的,只有在真正遍历结果范围的时候,才会执行各个适配器的逻辑,不会提前生成临时容器,内存效率很高。
  • 不是所有的范围操作都支持管道,只有标准库中标注为范围适配器的类型才支持通过|运算符组合。
  • 管道的组合顺序是从左到右依次执行,和Linux管道的执行逻辑一致,前一个操作的输出作为后一个操作的输入。
  • 自定义适配器的时候,要确保重载的operator|的参数和返回值类型符合范围的要求,避免出现编译错误。

总结

std::ranges::pipe是C++20 ranges库中非常实用的特性,它让范围算法的组合变得像管道一样直观简洁,大幅提升了数据处理代码的可读性。无论是使用标准库自带的范围适配器,还是自定义适配器,都可以通过管道运算符实现链式调用,减少临时变量的使用,让逻辑更加清晰。开发者可以在日常的数据处理场景中多尝试使用这种写法,提升C++代码的开发效率。

std::ranges::pipeC++范围算法管道组合函数对象修改时间:2026-06-23 14:12:37

免责声明:​ 已尽一切努力确保本网站所含信息的准确性。网站内容多为原创整理与精心编撰,观点力求客观中立。本站旨在免费分享,内容仅供个人学习、研究或参考使用。若引用了第三方作品,版权归原作者所有。如内容涉及您的权益,请联系我们处理。
内容垂直聚焦
专注技术核心技术栏目,确保每篇文章深度聚焦于实用技能。从代码技巧到架构设计,为用户提供无干扰的纯技术知识沉淀,精准满足专业提升需求。
知识结构清晰
覆盖从开发到部署的全链路。AI、前端、编程、数据库、服务器、建站、系统层层递进,构建清晰学习路径,帮助用户系统化掌握开发与运维所需的核心技术。
深度技术解析
拒绝泛泛而谈,深入技术细节与实践难点。无论是数据库优化还是服务器配置,均结合真实场景与代码示例进行剖析,致力于提供可直接应用于工作的解决方案。
专业领域覆盖
精准对应开发生命周期。从前端界面到后端编程,从数据库操作到服务器运维,形成完整闭环,一站式满足全栈工程师和运维人员的技术需求。
即学即用高效
内容强调实操性,步骤清晰、代码完整。用户可根据教程直接复现和应用于自身项目,显著缩短从学习到实践的距离,快速解决开发中的具体问题。
持续更新保障
专注既定技术方向进行长期、稳定的内容输出。确保各栏目技术文章持续更新迭代,紧跟主流技术发展趋势,为用户提供经久不衰的学习价值。