Modern C++ std::get<n>(tuple)的原理

本文详细讲解了std::tuple的取值方法,包括不同类型的处理,以及std::unique_ptr的size问题。通过实验和代码实现展示了std::tuple内存布局,还涉及了GDB调试技巧。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1. 前言

前面我们讲过std::tuple的实现原理,但没有讲如何取出数据,本节着重讲讲这点。本节与之前的blog有较大关联,如果您没看,这里有链接,链接已按由浅入深排好序,您可以按顺序阅读。如果时间少可以直接看第三篇由工具快速理解std::tuple的实现原理:

  1. Modern C++ std::unique_ptr的实现原理 带出size问题
  2. Modern C++ std::tuple的size 编程实验size问题
  3. Modern C++利用工具快速理解std::tuple的实现原理 快速理解std::tuple的实现原理
  4. GDB调试技巧实战–自动化画出类关系图 写一个普世工具结合GDB自动画类关系图,当然也适用std::tuple
  5. Modern C++ sizeof(std::tuple)的秘密及实现代码解读 从源码理解std::tuple的实现

2. 按图索骥

原理也不难,根据要取的数据是空类还是非空类分两种情况,先上图,懂了就不必往下看了。

请添加图片描述
实验代码贴在这,方便大家从copy试验:

#include<iostream>
#include <tuple>
using namespace std;

struct Empty{
        constexpr Empty() noexcept = default;
};

std::tuple<int,Empty,Empty,int> ie2i = {1, Empty(), Empty(), 2};//12
        auto& ie2i_0 = std::get<0>(ie2i);
        auto& ie2i_1 = std::get<1>(ie2i);
        auto& ie2i_2 = std::get<2>(ie2i);
        auto& ie2i_3 = std::get<3>(ie2i);
        std::cout<<"sizeof(std::tuple<int,Empty,Empty,int>):"<<sizeof(std::tuple<int,Empty,Empty,int>)<<std::endl;
        std::cout<<"ie2i addr:"<<&ie2i<<" ie2i_3:"<<&ie2i_3<<" ie2i_2:"<<&ie2i_2<<" ie2i_1:"<<&ie2i_1<<" ie2i_0:"<<&ie2i_0<<std::endl<<std::endl;

3. 看代码实现

3.1 先看取第一个元素:类型为int,值为1

在这里插入图片描述
注意下面由__t转换为__b, 即从类型std::_Tuple_impl<0, int, Empty, Empty, int> &转成了std::_Head_base<0, int, false> &,而且偏移加了8,说明在std::tuple<int,Empty, Empty, int>这个大房子里std::_Head_base<0, int, false> 位于+8处的偏房中。偏移大小和对象内存模型相关,我们不去讨论。
在这里插入图片描述

3.2 再看取第二个元素:类型为Empty:

在这里插入图片描述
地址从d4f0跳到了d4f4, 说明在std::tuple<int,Empty, Empty, int>这个大房子里std::_Head_base<1, Empty, true> 位于+4处的偏房中, 这一点可以由下面的反汇编代码证明():
在这里插入图片描述
$rax偷偷加了4传给了_M_head(_Head_base& __b)

/usr/include/c++/8/tuple
  72   template<std::size_t _Idx, typename _Head>
  73     struct _Head_base<_Idx, _Head, true>
  74     : public _Head
  75     {
... ...
 112       static constexpr _Head&
 113       _M_head(_Head_base& __b) noexcept { return __b; }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

深山老宅

鸡蛋不错的话,要不要激励下母鸡

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值