前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >C++17使用std::apply和fold expression对tuple进行遍历

C++17使用std::apply和fold expression对tuple进行遍历

作者头像
卡尔曼和玻尔兹曼谁曼
发布2019-01-22 09:30:30
1.9K0
发布2019-01-22 09:30:30
举报
文章被收录于专栏:给永远比拿愉快

C++17使用std::apply和fold expression对std::tuple进行遍历

std::apply函数

先来看这个std::apply函数,这个函数定义在tuple头文件中,函数签名如下:

代码语言:javascript
复制
template <class F, class Tuple>
constexpr decltype(auto) apply(F&& f, Tuple&& t);

该函数接受两个参数,第一个是一个函数对象,第二个是一个Tuple对象

来看一个最简单的示例:

代码语言:javascript
复制
#include <tuple>
#include <iostream>

int main() {
    // 两个元素相加
    std::cout << std::apply([](auto x, auto y) { return x + y; }, 
                            std::make_tuple(1, 2.0)) << '\n';
}

输出结果是3

这个例子中第一个参数使用Lambda匿名函数将tuple中的两个元素相加,第二个使用std::make_tuple函数构造一个只含有两个元素的tuple

fold expression

这个特性是C++ 17中我觉得很有用的一个新特性,使用规则有下面四条:

  1. Unary right fold (EEE opopop …) becomes (E1E_1E1​ opopop (… opopop (EN−1E_{N-1}EN−1​ opopop ENE_NEN​)))
  2. Unary left fold (… opopop EEE) becomes (((E1E_1E1​ opopop E2E_2E2​) opopop …) opopop ENE_NEN​)
  3. Binary right fold (EEE opopop … opopop III) becomes (E1E_1E1​ opopop (… opopop (EN−1E_{N−1}EN−1​ opopop (ENE_NEN​ opopop III))))
  4. Binary left fold (III opopop … opopop EEE) becomes ((((III opopop E1E_1E1​) opopop E2E_2E2​) opopop …) opopop ENE_NEN​)

这里的EEE指的是Expression(符合C++语法的表达式),opopop指的是operator(操作符),NNN是parameter pack(可变参数)的个数,III是一个常数。

可能看这个规则有些抽象,我们来看一些具体的例子:

代码语言:javascript
复制
#include <tuple>
#include <iostream>

int main() {
    // 多个元素相加,使用parameter pack
    std::cout << std::apply([](auto&& ... args) { return (args + ...); },
                            std::make_tuple(1, 2.f, 3.0)) << '\n';
    // 遍历tuple并输出,注意逗号操作符的使用
    std::apply([](auto&&... args) { ((std::cout << args << '\n'), ...); },
                std::make_tuple(1, 2.f, 3.0));
}

输出如下:

代码语言:javascript
复制
6
1
2
3

第6行中,std::apply函数的第一个参数是一个Lambda匿名函数,函数的参数是一个可变参数args,函数体中只有一条语句args + ...,这个情况就是上面的第一种情况:这里的EEE就是args,opopop就是+,所以展开来就是args1+args2+args3args_1 + args_2 + args_3args1​+args2​+args3​(因为参数的个数是3)。

第9行中,Lambda匿名函数的函数体是((std::cout << args << '\n'), ...)这是一个逗号操作符,也属于上面四种情况中的第一种:这里的EEE就是std::cout << args << '\n'),opopop就是,,所以这一行就打印输出了tuple的每一个元素。如果在C++17之前想要遍历tuple就比较麻烦,需要很多额外的操作。

参考资料

  1. std::apply
  2. fold expression
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2018年10月08日,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • C++17使用std::apply和fold expression对std::tuple进行遍历
    • std::apply函数
      • fold expression
        • 参考资料
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档