前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >【STL源码拆解】基于源码分析forward_lsit容器实现(详细!)

【STL源码拆解】基于源码分析forward_lsit容器实现(详细!)

作者头像
cpp加油站
发布2021-09-03 15:06:36
4530
发布2021-09-03 15:06:36
举报
文章被收录于专栏:cpp加油站cpp加油站

本篇文章介绍一下c++11中新增的顺序容器forward_list,基于stl的源码分析一下该容器的整体实现及数据结构。

说明一下,我用的是gcc7.1.0编译器,标准库源代码也是这个版本的。

按照惯例,还是先看一下本文大纲,如下:

1. forward_list是什么

forward_list是c++11为STL新增加的一种顺序容器,使用的时候包含头文件forward_list即可,真实的类声明位于头文件bits/forward_list.h中,类forward_list是一个类模板,基于单链表结构实现,下面我们就来基于forward_list的源码来看下它的具体实现。

2. forward_list周边类介绍

在正式开始介绍类模板forward_list之前,我们先了解下它所使用到的其他类型的介绍,这些类型是理解forward_list源码实现的前置条件。

2.1 类模板_Fwd_list_node和它的基类_Fwd_list_node_base

类模板_Fwd_list_node声明同样位于头文件bits/forward_list.h中,先看下类声明,如下:

代码语言:javascript
复制
template<typename _Tp>
    struct _Fwd_list_node
    : public _Fwd_list_node_base
    {
      _Fwd_list_node() = default;

      __gnu_cxx::__aligned_buffer<_Tp> _M_storage;

      _Tp*
      _M_valptr() noexcept
      { return _M_storage._M_ptr(); }

      const _Tp*
      _M_valptr() const noexcept
      { return _M_storage._M_ptr(); }
    };

该类定义了一个成员变量_M_storage,这个成员变量类型是__gnu_cxx::__aligned_buffer<_Tp>,同样是一个类模板,看看类模板__aligned_buffer的类声明,如下:

代码语言:javascript
复制
template<typename _Tp>
    struct __aligned_buffer
    : std::aligned_storage<sizeof(_Tp), std::alignment_of<_Tp>::value>
    {
      typename
 std::aligned_storage<sizeof(_Tp), std::alignment_of<_Tp>::value>::type _M_storage;
        ......  
 };

可以看到其实它是基于另外一个类aligned_storage实现的,且以这个基类中声明的类型type定义了一个同样的成员变量_M_storage,接着看类aligned_storage实现:

代码语言:javascript
复制
template<std::size_t _Len, std::size_t _Align =
    __alignof__(typename __aligned_storage_msa<_Len>::__type)>
    struct aligned_storage
    {
      union type
      {
 unsigned char __data[_Len];
 struct __attribute__((__aligned__((_Align)))) { } __align;
      };
    };

基于以上代码,可以看出类型aligned_storage::type其实是一个联合体类型,该联合体的第一个字段很明朗,就是以模板类型_Tp的长度定义了一个无符号字符数组,但第二个字段__align就比较奇怪了,他的类型是struct __attribute__((__aligned__((_Align)))) { },整体而言这是一个结构体,但大括号里面为空,说明该结构体是没有字段的,那么它到底是什么呢?

其实__attribute__是gcc的一个扩展用法,它允许在该关键字后面的双括号里面指定某些特殊属性,这里的__aligned__((_Align))就是指定了按多少个字节对齐,其中的_Align是通过非类型模板形参传入的对齐字节数。

我们在看下,这里的使用场景下类模板aligned_storage的实参是sizeof(_Tp), std::alignment_of<_Tp>::value,第一个实参就是模板类型_Tp的长度,第二个实参是基于alignment_of实现的,其实就是告诉编译器,根据模板类型_Tp的字节对齐规则去对齐。

所以简单来说,其实_Fwd_list_node::_M_storage就是一个结构体变量而已,它的大小是模板类型_Tp的大小。

接下来我们再看看基类_Fwd_list_node_base的实现,它的源码如下:

代码语言:javascript
复制
struct _Fwd_list_node_base
  {
    _Fwd_list_node_base() = default;

    _Fwd_list_node_base* _M_next = nullptr;
 ......

  };

这个类就比较简单哈,就是定义了一个成员指针而已,这里不再多说。

2.2 forward_list的基类_Fwd_list_base

forward_list的基类_Fwd_list_base声明同样位于头文件bits/forward_list.h中,同样的,先看看它有哪些成员变量,我们截取一段源码,如下

代码语言:javascript
复制
  template<typename _Tp, typename _Alloc>
    struct _Fwd_list_base
    {
    protected:
 typedef __alloc_rebind<_Alloc, _Tp>     _Tp_alloc_type;
      typedef __alloc_rebind<_Alloc, _Fwd_list_node<_Tp>> _Node_alloc_type;
      typedef __gnu_cxx::__alloc_traits<_Node_alloc_type> _Node_alloc_traits;
      struct _Fwd_list_impl 
      : public _Node_alloc_type
      {
        _Fwd_list_node_base _M_head;
       ......
      };
      _Fwd_list_impl _M_impl;
      ......
    };

_Fwd_list_base也是一个类模板,它只有一个成员变量_M_impl,该成员变量类型也是在这个类模板中定义的结构体类型_Fwd_list_impl,它的基类是_Node_alloc_type,这个类型比较复杂,但结合我们之前写的内存萃取器和内存分配器的说明,经过一番推导,可知这个基类真正的类型是allocator<_Fwd_list_node<_Tp>>,这是一个内存分配器,所以实际上_Fwd_list_impl它可以说是类模板_Fwd_list_base里面定义的一个内存萃取器,它根据模板类型取得一个内存分配器,只不过它多保存了一个成员变量_M_head而已,_M_head的类型我们在上一小节里面说了,这里不再多说。

3. forward_list框架实现

前面介绍了forward_list周边的一些类实现,接下来我们看看怎么把这些类与forward_list关联起来。

首先呢,我们还是来看一下forward_list的整体类图,如下:

基于以上类图以及上一章对于forward_list周边类的介绍,我们对类模板forward_list的整体类框架实现总结如下:

  • forward_list本身不保存数据,只实现了插入、删除、查询、构造等接口,供程序员们使用;
  • forward_list的基类是_Fwd_list_base,保存数据的成员变量_M_impl声明于该基类中,真正对数据的操作也都是基于该基类完成的;
  • 成员变量_M_impl中保存了头结点_M_head,头结点是链表的入口;
  • 头结点_M_head类型是_Fwd_list_node_base,该类型只包含一个成员变量_M_next_M_next是一个基类指针,它指向下一个链表节点,每个节点类型是_Fwd_list_node<_Tp>,该类型含两个成员变量,一个是继承于基类的_M_head,还有一个是该类本身定义的成员变量_M_storage

到这里,对forward_list我们就有了一些整体的认知了,但此时我们还是对于它的内存结构实现不是特别清晰,接下来就以forward_list的构造为一条线来详细看下最后它的内存结构是怎样的。

4. forward_list构造实现及内存结构

forward_list有很多种构造函数,包括拷贝构造、默认无参构造、有参构造、移动构造等,这里我们以其中一种有参构造为例,该构造函数声明如下:

代码语言:javascript
复制
//第一个参数为容器需构造的元素数量,第二个参数为每个元素的值,第三个参数是分配器,其中分配器是类的模板参数指定,这里我们使用默认的即可
forward_list(size_type __n, const _Tp& __value,
                   const _Alloc& __al = _Alloc())
      : _Base(_Node_alloc_type(__al))
      { _M_fill_initialize(__n, __value); }

该函数首先调用基类构造函数指定内存分配器,关于这一点我们使用默认的即可,这里不再多说。

然后调用了函数_M_fill_initialize进行动态内存申请及元素赋值等操作,该函数源码实现如下:

代码语言:javascript
复制
template<typename _Tp, typename _Alloc>
    void
    forward_list<_Tp, _Alloc>::
    _M_fill_initialize(size_type __n, const value_type& __value)
    {
        //头节点是类对象,地址固定,这里取头节点地址赋给临时指针__to
      _Node_base* __to = &this->_M_impl._M_head;
      for (; __n; --__n)
        {
          //创建一个节点,且将节点地址赋给上一节点的_M_next
          __to->_M_next = this->_M_create_node(__value);
          //__to指向上一行代码新创建的节点
          __to = __to->_M_next;
        }
    }

该函数就比较简单了,具体作用注释也写明了,我们接下来看看具体创建节点是怎么样的,节点创建使用函数_M_create_node,该函数实现在forward_list基类中,源码如下:

代码语言:javascript
复制
_Node* _M_get_node()
{
    //这里使用指定内存分配器申请一个_Fwd_list_node大小的空间,并返回指向该空间的地址给__ptr
 auto __ptr = _Node_alloc_traits::allocate(_M_get_Node_allocator(), 1);
    //获取__ptr的地址并返回
 return std::__addressof(*__ptr);
}
//这里使用了变参数模板,关于变参数模板的详细说明,我在上一篇文章中详细说明了,这里不再多说
template<typename... _Args>
_Node* _M_create_node(_Args&&... __args)
{
    _Node* __node = this->_M_get_node();
    __try
      {
    _Tp_alloc_type __a(_M_get_Node_allocator());
    typedef allocator_traits<_Tp_alloc_type> _Alloc_traits;
        //这里是placement new的用法,做了二次分配
    ::new ((void*)__node) _Node;
        //这里会根据函数入参调用元素类型的构造函数并进行赋值
    _Alloc_traits::construct(__a, __node->_M_valptr(),
               std::forward<_Args>(__args)...);
      }
    __catch(...)
      {
        this->_M_put_node(__node);
        __throw_exception_again;
      }
    return __node;
}

详细的说明,在上面代码的注释中都写明了,那么根据构造函数的调用路线,我们可以画出forward_list内存结构如下:

这是一个典型的头节点不存储数据的单链表结构,这里就不再多说啦。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2021-08-06,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 cpp加油站 微信公众号,前往查看

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

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

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1. forward_list是什么
  • 2. forward_list周边类介绍
    • 2.1 类模板_Fwd_list_node和它的基类_Fwd_list_node_base
      • 2.2 forward_list的基类_Fwd_list_base
      • 3. forward_list框架实现
      • 4. forward_list构造实现及内存结构
      相关产品与服务
      容器服务
      腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档