前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >C++ STL源码剖析 tr1与std array

C++ STL源码剖析 tr1与std array

作者头像
公众号guangcity
发布2019-10-15 16:42:54
1.1K0
发布2019-10-15 16:42:54
举报
文章被收录于专栏:光城(guangcity)光城(guangcity)

C++ STL源码剖析 tr1与std array

0.导语

源码剖析版本为gcc4.9.1。

C++ tr1全称Technical Report 1,是针对C++标准库的第一次扩展。即将到来的下一个版本的C++标准c++0x会包括它,以及一些语言本身的扩充。tr1包括大家期待已久的smart pointer,正则表达式以及其他一些支持范型编程的内容。草案阶段,新增的类和模板的名字空间是std::tr1。

1.std::tr1::array

使用:

代码语言:javascript
复制
#include <tr1/array>
std::tr1::array<int ,10> a;

tr1中的array比较简单,模拟语言本身的数组,并且让其支持迭代器操作,使其同其他容器一样,能够调用算法。对于tr1中array没有构造与析构。迭代器是直接使用传递进来的类型定义指针。

简单的看一下这个静态数组array源码:

代码语言:javascript
复制
template<typename _Tp, std::size_t _Nm>
struct array
{
    typedef _Tp 	    			      value_type;
    typedef value_type&                   	      reference;
    typedef const value_type&             	      const_reference;
    typedef value_type*          		      iterator;
    typedef const value_type*			      const_iterator;
    typedef std::size_t                    	      size_type;
    typedef std::ptrdiff_t                   	   difference_type;
    typedef std::reverse_iterator<iterator>	      reverse_iterator;
    typedef std::reverse_iterator<const_iterator>   const_reverse_iterator;
}

里面使用reverse_iterator作为rbegin与rend操作的迭代器。看上去上面一个迭代器,实际上两个,还有一个iterator,这个直接使用传递进来的类型定义指针,作为迭代器。

可以将其对比为vector中的正向与反向迭代器。

值得注意的是,在tr1::array中,支持传递数组大小为0,例如我们使用如下:

代码语言:javascript
复制
std::tr1::array<int,0> a;

对于这样的写法,会对应到下面:

代码语言:javascript
复制
// Support for zero-sized arrays mandatory.
value_type _M_instance[_Nm ? _Nm : 1];

根据传递进来的大小,如果不为0,就是传递进来的大小,否则为1。

2.std::array

使用

代码语言:javascript
复制
std::array<int ,10> a;

std中的array包含了

std_array.png

对比tr1与std的array

代码语言:javascript
复制
template<typename _Tp, std::size_t _Nm>
struct array
{
    typedef _Tp 	    			      value_type;
    typedef value_type*			      pointer;
    typedef const value_type*                       const_pointer;
    typedef value_type&                   	      reference;
    typedef const value_type&             	      const_reference;
    typedef value_type*          		      iterator;
    typedef const value_type*			      const_iterator;
    typedef std::size_t                    	      size_type;
    typedef std::ptrdiff_t                   	      difference_type;
    typedef std::reverse_iterator<iterator>	      reverse_iterator;
    typedef std::reverse_iterator<const_iterator>   const_reverse_iterator;

    // Support for zero-sized arrays mandatory.
    typedef _GLIBCXX_STD_C::__array_traits<_Tp, _Nm> _AT_Type;    // # define _GLIBCXX_STD_C std
    typename _AT_Type::_Type                         _M_elems;
}

发现array里面有两处值得注意的地方:

代码语言:javascript
复制
// Support for zero-sized arrays mandatory.
typedef _GLIBCXX_STD_C::__array_traits<_Tp, _Nm> _AT_Type;    // # define _GLIBCXX_STD_C std
typename _AT_Type::_Type                         _M_elems;

在源码中去找__array_traits,看到:

代码语言:javascript
复制
template<typename _Tp, std::size_t _Nm>
struct __array_traits
{
    typedef _Tp _Type[_Nm];

    static constexpr _Tp&
    _S_ref(const _Type& __t, std::size_t __n) noexcept
    { return const_cast<_Tp&>(__t[__n]); }
};

上面两行的代码可以理解为下面:

代码语言:javascript
复制
typedef _Tp _Type[100];
typedef _Type _M_elems;  // 一个含有100个元素的数组。

在实际写代码的时候,如果要定义一个数组,我们可以这样写:

代码语言:javascript
复制
int a[100];
//或者
typedef int T[100];
typedef T a;

针对传进来的size处理,相比于tr1,更加复杂,使用了模板偏特化来处理传递size为0情况。

代码语言:javascript
复制
template<typename _Tp, std::size_t _Nm>
struct __array_traits
{
    typedef _Tp _Type[_Nm];

    static constexpr _Tp&
    _S_ref(const _Type& __t, std::size_t __n) noexcept
    { return const_cast<_Tp&>(__t[__n]); }
};

template<typename _Tp>
struct __array_traits<_Tp, 0>
{
    struct _Type { };

    static constexpr _Tp&
    _S_ref(const _Type&, std::size_t) noexcept
    { return *static_cast<_Tp*>(nullptr); }
};
本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-10-15,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 光城 微信公众号,前往查看

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

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

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • C++ STL源码剖析 tr1与std array
    • 1.std::tr1::array
      • 2.std::array
      相关产品与服务
      容器服务
      腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档