C++ STL list

简介: 上次我们详细的介绍了vector,今天我们继续来介绍一下TSTL中的另外一个容器list。list在基础的功能和结构上就是一个双向带头的循环链表,实现起来基本不难,但是list迭代器的封装是非常值得学习的。

 

✅<1>主页:我的代码爱吃辣

📃<2>知识讲解:C++之 STL list介绍和模拟实现

☂️<3>开发环境:Visual Studio 2022

💬<4>前言:上次我们详细的介绍了vector,今天我们继续来介绍一下TSTL中的另外一个容器list。list在基础的功能和结构上就是一个双向带头的循环链表,实现起来基本不难,但是list迭代器的封装是非常值得学习的。

目录

一.认识list

二.list的使用

1.构造函数

2.增删查改

3.list 迭代器

四.list 模拟实现

1.链表结点

2.list整体结构

3.list构造函数

4.push_back(),pop_back()

5.迭代器

5.1迭代器结构

5.2迭代器操作

5.2.1operator*()

5.2.2 operator ++(),operator --()

5.2.3operator==(),operator!=()

5.2.4operator->()

5.2.5begin(),end()

5.3 const 迭代器

5.4 insert(),erase()

5.5 析构函数

5.6再看构造函数

五.list模拟实现整体代码


image.gif编辑

一.认识list

list - C++ Reference

    1. list是可以在常数范围内在任意位置进行插入和删除的序列式容器,并且该容器可以前后双向迭代。
    2. list的底层是双向链表结构,双向链表中每个元素存储在互不相关的独立节点中,在节点中通过指针指向其前一个元素和后一个元素。
    3. list与forward_list非常相似:最主要的不同在于forward_list是单链表,只能朝前迭代,已让其更简单高效。
    4. 与其他的序列式容器相比(array,vector,deque),list通常在任意位置进行插入、移除元素的执行效率更好。
    5. 与其他序列式容器相比,list和forward_list最大的缺陷是不支持任意位置的随机访问,比如:要访问list的第6个元素,必须从已知的位置(比如头部或者尾部)迭代到该位置,在这段位置上迭代需要线性的时间开销;list还需要一些额外的空间,以保存每个节点的相关联信息(对于存储类型较小元素的大list来说这可能是一个重要的因素)。

    结构:list使用的是双向的循环带头结点的链表。

    image.gif编辑

    二.list的使用

    list的使用非常简单,有了我们之前vector和string的基础。上手list基本就是小菜一碟。

    1.构造函数

    构造函数( (constructor)) 接口说明
    list (size_type n, const value_type& val = value_type()) 构造的list中包含n个值为val的元素
    list() 构造空的list
    list (const list& x) 拷贝构造函数
    list (InputIterator first, InputIterator last) 用[first, last)区间中的元素构造list

    2.增删查改

    list<int> li;
      //尾插
      li.push_back(10);
      //头插
      li.push_front(20);
      //尾删
      li.pop_back();
      //头删
      li.pop_front();
      //迭代器位置插入数据
      li.insert(li.begin(), 100);
      //删除迭代器位置数据
      li.erase(li.begin());

    image.gif

    3.list 迭代器

    函数声明 接口说明
    begin +end 返回第一个元素的迭代器+返回最后一个元素下一个位置的迭代器
    rbegin +
    rend
    返回第一个元素的reverse_iterator,即end位置,返回最后一个元素下一个位置的
    reverse_iterator,即begin位置

    list的迭代器使用与string和vector一模一样,在此不多介绍。

    四.list 模拟实现

    1.链表结点

    template<class T>
    struct _list_node
    {
      _list_node( const T& val = T())
        :_val(val),
        _next(nullptr),
        _prev(nullptr)
      {
      }
      T _val;           //存储数据
      _list_node* _next;//后一个结点的指针
      _list_node* _prev;//前一个结点的指针
    };

    image.gif

    注意:我们在此处的struct定义了一个类,我们在定义_next和_prev的时候就不用像C语言那样加上struct。

    例如:C语言写法

    template<class T>
    struct _list_node
    {
      _list_node(const T& val = T())
        :_val(val),
        _next(nullptr),
        _prev(nullptr)
      {
      }
      T _val;           //存储数据
      struct _list_node* _next;//后一个结点的指针
      struct _list_node* _prev;//前一个结点的指针
    };

    image.gif

    2.list整体结构

    template<class T>
    class List
    {
    public:
        typedef _list_node<T> node;
        //...成员方法 增删查改
    private:
      node* _head;//头节点
    };

    image.gif

    3.list构造函数

    因为我们的list是一个带哨兵位的双向循环链表。所以这里我们将new出来的结点,设置为循环双向结构。

    List()
        :_head(new node)
      {
        _head->_next = _head;
        _head->_prev = _head;
      }

    image.gif

    4.push_back(),pop_back()

    push_back()尾插一个数据,pop_back()尾删一个数据。

    //push_back
        void push_back(const T& val )
      {
            //创建结点
        node* newnode = new node(val);
        node* tail = _head->_prev;
            //改变指向
        tail->_next = newnode;
        newnode->_next = _head;
        _head->_prev = newnode;
        newnode->_prev = tail;
      }

    image.gif

    //pop_back
        void pop_back()
      {
        //判空
        assert(!empty());
        node* tail = _head->_prev;
        node* newtail = tail->_prev;
        //改变指向
        _head->_next = newtail;
        newtail->_prev = _head;
        //释放结点
        delete tail;
      }

    image.gif

    image.gif编辑

    5.迭代器

    list的迭代器不同于string和vector,因为list的物理存储是不连续的,所以我们不能像list和vector一样仅仅使用原生指针来作为list的迭代器。

    迭代器的实际就是一种模仿指针的行为来为容器提供一种通用的访问方法的设计。虽然list的迭代器不能使用原生指针来替代,但是我们可以对原生的指针进行类级别的封装来构造迭代器,并且使得迭代器具有指针的行为。

    迭代器有两种实现方式,具体应根据容器底层数据结构实现:

      1.  原生态指针,比如:vector
      2.  将原生态指针进行封装,因迭代器使用形式与指针完全相同,因此在自定义的类中必须实现以下方法:
        • 指针可以解引用,迭代器的类中必须重载operator*()
        • 指针可以通过->访问其所指空间成员,迭代器类中必须重载oprator->()
        • 指针可以++向后移动,迭代器类中必须重载operator++()与operator++(int)
        • 至于operator--()/operator--(int)释放需要重载,根据具体的结构来抉择,双向链表可以向前移动,所以需要重载,如果是forward_list就不需要重载--
        • 迭代器需要进行是否相等的比较,因此还需要重载operator==()与operator!=()

        5.1迭代器结构

        迭代器的整体结构如下。

        template<class T>
        class _list_iterator
        {
        public:
          typedef _list_node<T> node;    //结点类
          typedef _list_iterator<T> self;//迭代器本身
            //使用原生指针构造迭代器
          _list_iterator(node* n)
            :_node(n)
          {
          }
            //operator*()
            //operator++()
            //operator--()
            //operator->()
            //operator==()
            //operator!=()
        private:
          node* _node;//迭代器是对原生指针的封装
        };

        image.gif

        5.2迭代器操作

        5.2.1operator*()

        T& operator*()
          {
            return _node->_val;
          }

        image.gif

        返回该结点的值的引用。

        5.2.2 operator ++(),operator --()

        ++的操作其实就是让迭代器挪到下一个位置,--操作其实就是让迭代器挪到上一个位置。迭代器++或者--操作的返回值还是一个迭代器。

        //后置++
          self operator++(int)
          {
            self tmp(*this);
            _node = _node->_next;
            return tmp;
          }
            //前置++
          self operator++()
          {
            _node = _node->_next;
            return *this;
          }
            //前置--
            self operator--()
          {
            _node = _node->_prev;
            return *this;
          }
            //后置--
            self operator--(int)
          {
            self tmp(*this);
            _node = _node->_prev;
            return tmp;
          }

        image.gif

        5.2.3operator==(),operator!=()

        比较两个迭代器是否相等直接比较迭代器封装的两个指针是否相等就可以了。

        bool operator!=(const self& it)
          {
            return _node != it._node;
          }
          bool operator==(const self& it)
          {
            return _node == it._node;
          }

        image.gif

        5.2.4operator->()

        我们知道一般的结构体,类指针都是支持使用->访问一些成员的。当list存储的数据是结构体,或者是类试迭代器也应该支持->去访问。

        T*  operator->()
          {
            return &_node->_val;
          }

        image.gif

        那么这样我们在外边的访问会不会很奇怪呢?

        struct AA
            {
              AA(int aa = 10)
                :_aa(aa)
              {
              }
                int _aa;
            };
            list<AA> lli(10, AA()); 
            list<AA>::iterator lit = lli.begin();
            //1.调用:
            lit.operator->()->_aa;
            //2.调用:
            lit->_aa;

        image.gif

        image.gif编辑

        在实际调用的时候我们可以直接写成调用2。

        5.2.5begin(),end()

        typedef _list_iterator<T> iterator;
            iterator begin()
          {
            //使用哨兵位的下一个结点指针,构造begin
            return iterator(_head->_next);
          }
          iterator end()
          {
            //使用哨兵位结点指针,构造end
            return iterator(_head);
          }

        image.gif

        可以构成 [ begin,end ).这种左闭右开的结构。

        5.3 const 迭代器

        const迭代器与普通迭代器最大的不同,就是const迭代器不允许修改迭代器指向的数据。在整个迭代器操作中只有,operator* 和 operator->是对指向数据操作的。我们仅需要将普通迭代器的这两个函数返回值修改即可。并且区分开普通迭代器与const迭代器。

        template<class T>
        class const_list_iterator
        {
        public:
          typedef _list_node<T> node;
          typedef const_list_iterator self;
          const_list_iterator(node* n)
            :_node(n)
          {
          }
            //返回值加const,不允许修改
          const T& operator*()
          {
            return _node->_val;
          }
          self operator++(int)
          {
            self tmp(*this);
            _node = _node->_next;
            return tmp;
          }
          self operator++()
          {
            _node = _node->_next;
            return *this;
          }
          self operator--(int)
          {
            self tmp(*this);
            _node = _node->_prev;
            return tmp;
          }
          self operator--()
          {
            _node = _node->_prev;
            return *this;
          }
          bool operator!=(const self& it)
          {
            return _node != it._node;
          }
          bool operator==(const self& it)
          {
            return _node == it._node;
          }
            //返回值加const,不允许修改
          const T*  operator->()
          {
            return &_node->_val;
          }
        private:
          node* _node;
        };

        image.gif

        但是这种设计让人觉得很不舒服,代码的重复太多。从代码的角度来看,const和普通迭代器仅仅只是返回值不同而已。我们只要给在给迭代器不同的类型,我们只需要给模板多架两个参数,如果是 T& 和 T*就实例化出普通迭代器,如果是 const T& 和 const T*就实例化出const迭代器。

        list类内部:

        template<class T>
        class List
        {
        public:
          typedef _list_node<T> node;
          typedef _list_iterator<T,T&,T*> iterator;//普通迭代器
          typedef _list_iterator<T,const T&,const T*> const_iterator;//const迭代器
          List()
            :_head(new node)
          {
            _head->_next = _head;
            _head->_prev = _head;
          }
          iterator begin()
          {
            //使用哨兵位的下一个结点指针,构造begin
            return iterator(_head->_next);
          }
          iterator end()
          {
            //使用哨兵位结点指针,构造end
            return iterator(_head);
          }
          const_iterator begin()const
          {
            return const_iterator(_head->_next);
          }
          const_iterator end()const
          {
            return const_iterator(_head);
          }
        private:
          node* _head;
        };

        image.gif

        迭代器:

        //T:数据类型,Ref:T&/const T&, Ptr: T*/const T*
        template<class T,class Ref,class Ptr>
        class _list_iterator
        {
        public:
          typedef _list_node<T> node;
          typedef _list_iterator<T,Ref,Ptr> self;
          _list_iterator(node* n)
            :_node(n)
          {
          }
          Ref operator*()
          {
            return _node->_val;
          }
          self operator++(int)
          {
            self tmp(*this);
            _node = _node->_next;
            return tmp;
          }
          Ptr operator->()
          {
            return &_node->_val;
          }
          bool operator!=(const self& it )
          {
            return _node != it._node;
          }
          bool operator==(const self& it)
          {
            return _node == it._node;
          }
            //返回迭代器内部的结点指针
          node* GetNodePtr()
          {
            return _node;
          }
        private:
          node* _node;
        };

        image.gif

        根据不同的模板参数从而实例化出不同的代码。

        5.4 insert(),erase()

        insert()可以在某一迭代器位置插入数据,erase可以删除某一迭代器位置的数据。

        void insert(iterator pos,const T& val)
          {
                //创建结点
            node* newnode = new node(val);
            node* posnode = pos.GetNodePtr();
            node* prevnode = posnode->_prev;
                //改变指向
            newnode->_next = posnode;
            newnode->_prev = prevnode;
            prevnode->_next = posnode;
            posnode->_prev = newnode;
          }
            iterator erase(iterator pos)
          {
            node* posnode = pos.GetNodePtr();
            node* prevnode = posnode->_prev;
            node* nextnode = posnode->_next;
                //修改指向
            prevnode->_next = nextnode;
            nextnode->_prev = prevnode;
                //释放结点
            delete posnode;
                return iterator(nextnode);
          }

        image.gif

        erase在返回的删除位置的下一个结点的迭代器,也是为了绝迭代器失效的问题。

        5.5 析构函数

        //清除数据,但是头节点需要保留
          void clear()
          {
            iterator it = begin();
            while (it != end())
            {
              erase(it++);
            }
          }
          ~List()
          {
            //连同头节点一起释放
            clear();
            delete _head;
            _head = nullptr;
          }

        image.gif

        5.6再看构造函数

        迭代器初始化

        void emptyInit()
          {
            _head->_next = _head;
            _head->_prev = _head;
          }
            template<class InputIterator>
          List(InputIterator frist, InputIterator lest)
            :_head(new node)
          {
            emptyInit();
            while (frist != lest)
            {
              push_back(*frist);
              frist++;
            }
          }

        image.gif

        五.list模拟实现整体代码

        #pragma once
        #include<iostream>
        #include<cassert>
        using namespace std;
        template<class T>
        struct _list_node
        {
          _list_node( const T& val = T())
            :_val(val),
            _next(nullptr),
            _prev(nullptr)
          {
          }
          T _val;           //存储数据
          _list_node* _next;//后一个结点的指针
          _list_node* _prev;//前一个结点的指针
        };
        /*
        List 的迭代器
        迭代器有两种实现方式,具体应根据容器底层数据结构实现:
          1. 原生态指针,比如:vector
          2. 将原生态指针进行封装,因迭代器使用形式与指针完全相同,因此在自定义的类中必须实现以下方法:
           1. 指针可以解引用,迭代器的类中必须重载operator*()
           2. 指针可以通过->访问其所指空间成员,迭代器类中必须重载oprator->()
           3. 指针可以++向后移动,迭代器类中必须重载operator++()与operator++(int)
            至于operator--()/operator--(int)释放需要重载,根据具体的结构来抉择,双向链表可以向前移动,所以需要重载,如果是forward_list就不需要重载--
           4. 迭代器需要进行是否相等的比较,因此还需要重载operator==()与operator!=()
        */
        //普通迭代器1.0版本
        //template<class T>
        //class _list_iterator
        //{
        //public:
        //  typedef _list_node<T> node;
        //  typedef _list_iterator self;
        //
        //  _list_iterator(node* n)
        //    :_node(n)
        //  {
        //  }
        //
        //  T& operator*()
        //  {
        //    return _node->_val;
        //  }
        //
        //  self operator++(int)
        //  {
        //    self tmp(*this);
        //    _node = _node->_next;
        //    return tmp;
        //  }
        //
        //  self operator++()
        //  {
        //    _node = _node->_next;
        //    return *this;
        //  }
        //
        //  self operator--(int)
        //  {
        //    self tmp(*this);
        //    _node = _node->_prev;
        //    return tmp;
        //  }
        //
        //  self operator--()
        //  {
        //    _node = _node->_prev;
        //    return *this;
        //  }
        //
        //  bool operator!=(const self& it)
        //  {
        //    return _node != it._node;
        //  }
        //
        //  bool operator==(const self& it)
        //  {
        //    return _node == it._node;
        //  }
        //
        //  T*  operator->()
        //  {
        //    return &_node->_val;
        //  }
        //private:
        //  node* _node;
        //};
        //const迭代器1.0版本
        //template<class T>
        //class const_list_iterator
        //{
        //public:
        //  typedef _list_node<T> node;
        //  typedef const_list_iterator self;
        //
        //  const_list_iterator(node* n)
        //    :_node(n)
        //  {
        //  }
        //
        //  const T& operator*()
        //  {
        //    return _node->_val;
        //  }
        //
        //  self operator++(int)
        //  {
        //    self tmp(*this);
        //    _node = _node->_next;
        //    return tmp;
        //  }
        //
        //  self operator++()
        //  {
        //    _node = _node->_next;
        //    return *this;
        //  }
        //
        //  self operator--(int)
        //  {
        //    self tmp(*this);
        //    _node = _node->_prev;
        //    return tmp;
        //  }
        //
        //  self operator--()
        //  {
        //    _node = _node->_prev;
        //    return *this;
        //  }
        //
        //  bool operator!=(const self& it)
        //  {
        //    return _node != it._node;
        //  }
        //
        //  bool operator==(const self& it)
        //  {
        //    return _node == it._node;
        //  }
        //
        //  const T*  operator->()
        //  {
        //    return &_node->_val;
        //  }
        //private:
        //  node* _node;
        //};
        //迭代器2.0版本
        //T:数据类型,Ref:T&/const T&, Ptr: T*/const T*
        template<class T, class Ref, class Ptr>
        class _list_iterator
        {
        public:
          typedef _list_node<T> node;
          typedef _list_iterator<T, Ref, Ptr> self;
          _list_iterator(node* n)
            :_node(n)
          {
          }
          Ref operator*()
          {
            return _node->_val;
          }
          self operator++(int)
          {
            self tmp(*this);
            _node = _node->_next;
            return tmp;
          }
          Ptr operator->()
          {
            return &_node->_val;
          }
          bool operator!=(const self& it)
          {
            return _node != it._node;
          }
          bool operator==(const self& it)
          {
            return _node == it._node;
          }
          node* GetNodePtr()
          {
            return _node;
          }
        private:
          node* _node;
        };
        template<class T>
        class List
        {
        public:
          typedef _list_node<T> node;
          typedef _list_iterator<T,T&,T*> iterator;
          typedef _list_iterator<T,const T&,const T*> const_iterator;
          void emptyInit()
          {
            _head->_next = _head;
            _head->_prev = _head;
          }
          List()
            :_head(new node)
          {
            emptyInit();
          }
          template<class InputIterator>
          List(InputIterator frist, InputIterator lest)
            :_head(new node)
          {
            emptyInit();
            while (frist != lest)
            {
              push_back(*frist);
              frist++;
            }
          }
          void push_back(const T& val )
          {
            node* newnode = new node(val);
            node* tail = _head->_prev;
            tail->_next = newnode;
            newnode->_next = _head;
            _head->_prev = newnode;
            newnode->_prev = tail;
          }
          bool empty()
          {
            return _head->_next == _head;
          }
          void pop_back()
          {
            //判空
            assert(!empty());
            node* tail = _head->_prev;
            node* newtail = tail->_prev;
            //改变指向
            _head->_next = newtail;
            newtail->_prev = _head;
            //释放结点
            delete tail;
          }
          void insert(iterator pos,const T& val)
          {
            node* newnode = new node(val);
            node* posnode = pos.GetNodePtr();
            node* prevnode = posnode->_prev;
            newnode->_next = posnode;
            newnode->_prev = prevnode;
            prevnode->_next = posnode;
            posnode->_prev = newnode;
          }
          iterator erase(iterator pos)
          {
            node* posnode = pos.GetNodePtr();
            node* prevnode = posnode->_prev;
            node* nextnode = posnode->_next;
            prevnode->_next = nextnode;
            nextnode->_prev = prevnode;
            delete posnode;
            return iterator(nextnode);
          }
          iterator begin()
          {
            //使用哨兵位的下一个结点指针,构造begin
            return iterator(_head->_next);
          }
          iterator end()
          {
            //使用哨兵位结点指针,构造end
            return iterator(_head);
          }
          const_iterator begin()const
          {
            return const_iterator(_head->_next);
          }
          const_iterator end()const
          {
            return const_iterator(_head);
          }
          //清除数据,但是需要保留
          void clear()
          {
            iterator it = begin();
            while (it != end())
            {
              it = erase(it);
            }
          }
          ~List()
          {
            //连同头节点一起释放
            clear();
            delete _head;
            _head = nullptr;
          }
        private:
          node* _head;
        };

        image.gif

        相关文章
        |
        21天前
        |
        编译器 C语言 C++
        【c++丨STL】list模拟实现(附源码)
        本文介绍了如何模拟实现C++中的`list`容器。`list`底层采用双向带头循环链表结构,相较于`vector`和`string`更为复杂。文章首先回顾了`list`的基本结构和常用接口,然后详细讲解了节点、迭代器及容器的实现过程。 最终,通过这些步骤,我们成功模拟实现了`list`容器的功能。文章最后提供了完整的代码实现,并简要总结了实现过程中的关键点。 如果你对双向链表或`list`的底层实现感兴趣,建议先掌握相关基础知识后再阅读本文,以便更好地理解内容。
        25 1
        |
        1月前
        |
        算法 C语言 C++
        【c++丨STL】list的使用
        本文介绍了STL容器`list`的使用方法及其主要功能。`list`是一种双向链表结构,适用于频繁的插入和删除操作。文章详细讲解了`list`的构造函数、析构函数、赋值重载、迭代器、容量接口、元素访问接口、增删查改操作以及一些特有的操作接口如`splice`、`remove_if`、`unique`、`merge`、`sort`和`reverse`。通过示例代码,读者可以更好地理解如何使用这些接口。最后,作者总结了`list`的特点和适用场景,并预告了后续关于`list`模拟实现的文章。
        51 7
        |
        1月前
        |
        存储 编译器 C++
        C++ initializer_list&&类型推导
        在 C++ 中,`initializer_list` 提供了一种方便的方式来初始化容器和传递参数,而右值引用则是实现高效资源管理和移动语义的关键特性。尽管在实际应用中 `initializer_list&&` 并不常见,但理解其类型推导和使用方式有助于深入掌握现代 C++ 的高级特性。
        23 4
        |
        2月前
        |
        存储 编译器 C语言
        【c++丨STL】vector的使用
        本文介绍了C++ STL中的`vector`容器,包括其基本概念、主要接口及其使用方法。`vector`是一种动态数组,能够根据需要自动调整大小,提供了丰富的操作接口,如增删查改等。文章详细解释了`vector`的构造函数、赋值运算符、容量接口、迭代器接口、元素访问接口以及一些常用的增删操作函数。最后,还展示了如何使用`vector`创建字符串数组,体现了`vector`在实际编程中的灵活性和实用性。
        96 4
        |
        2月前
        |
        C语言 C++ 容器
        【c++丨STL】string模拟实现(附源码)
        本文详细介绍了如何模拟实现C++ STL中的`string`类,包括其构造函数、拷贝构造、赋值重载、析构函数等基本功能,以及字符串的插入、删除、查找、比较等操作。文章还展示了如何实现输入输出流操作符,使自定义的`string`类能够方便地与`cin`和`cout`配合使用。通过这些实现,读者不仅能加深对`string`类的理解,还能提升对C++编程技巧的掌握。
        100 5
        |
        2月前
        |
        存储 编译器 C语言
        【c++丨STL】string类的使用
        本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
        77 2
        |
        2月前
        |
        存储 算法 Linux
        【c++】STL简介
        本文介绍了C++标准模板库(STL)的基本概念、组成部分及学习方法,强调了STL在提高编程效率和代码复用性方面的重要性。文章详细解析了STL的六大组件:容器、算法、迭代器、仿函数、配接器和空间配置器,并提出了学习STL的三个层次,旨在帮助读者深入理解和掌握STL。
        79 0
        |
        1月前
        |
        存储 编译器 C语言
        【c++丨STL】vector模拟实现
        本文深入探讨了 `vector` 的底层实现原理,并尝试模拟实现其结构及常用接口。首先介绍了 `vector` 的底层是动态顺序表,使用三个迭代器(指针)来维护数组,分别为 `start`、`finish` 和 `end_of_storage`。接着详细讲解了如何实现 `vector` 的各种构造函数、析构函数、容量接口、迭代器接口、插入和删除操作等。最后提供了完整的模拟实现代码,帮助读者更好地理解和掌握 `vector` 的实现细节。
        46 0
        |
        11天前
        |
        C++ 芯片
        【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
        声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
        51 18
        |
        11天前
        |
        存储 编译器 数据安全/隐私保护
        【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
        声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
        37 13

        热门文章

        最新文章