【C++STL精讲】vector的模拟实现

简介: 【C++STL精讲】vector的模拟实现

0000000000000000000000000000000000000000000000000000000.png

目录


定义vector类

各成员函数的实现

构造函数

迭代器

size与capacity——求大小与容量

reserve——扩容

关于reserve中的深浅拷贝问题

resize——扩容并初始化

push_back——尾插

pop_back——尾删

insert——插入

erase——删除

empty——判空

[]重载——访问元素

传值构造

迭代器区间构造

赋值重载

拷贝构造

拷贝构造中的深浅拷贝问题

析构函数

完整源码


文章导读


本章我们将参照STL源码来模拟实现vector,这要求我们具备数据结构的基础且了解vector的基本使用。模式实现vector,将锻炼我们的代码能力,加深对类和对象的认识,同时能使我们对vector的使用更加游刃有余。


正文


定义vector类


为了区别于标准库中的vector,我们可以使用自己的命名空间,在自己的命名空间中模拟实现vector。我们已经了解过库中vector的基本使用,知道vector是一个可以存储任何类型的容器,为了实现各种类型都可以匹配,我们可以利用模板来实现。


STL源码中,vector包含三个基本成员:

  • iterator _start:指向首元素的迭代器;
  • iterator _finish:指向尾元素下一位的迭代器;
  • iterator _end_of_storage:指向最大容量的下一位的迭代器;


我们可以把迭代器理解为像指针一样的东西。

namespace hxy
{
  template<class T>
  class vector
  {
  public:
    typedef T* iterator;
    typedef const T* const_iterator;
  private:
    iterator _start;
    iterator _finish;
    iterator _end_of_storage;
  };
}


各成员函数的实现


构造函数


  vector()
    :_start(nullptr),
    _finish(nullptr),
    _end_of_storage(nullptr)
    {}


迭代器


  iterator begin()
  {
    return _start;
  }
  iterator end()
  {
    return _finish;
  }
  const_iterator begin() const
  {
    return _start;
  }
  const_iterator end() const
  {
    return _finish;
  }


size与capacity——求大小与容量


  size_t size() const
  {
    return _finish - _start; //指针相减即为个数
  }
  size_t capacity() const
  {
    return _end_of_storage - _start;
  }


reserve——扩容


  void reserve(size_t n)
  {
    if (n > capacity())
    {
      size_t sz = size();
      //使用中间变量,防止new失败
      iterator tmp = new T[n];
      if (_start)
      {
        //转移数据
        for (size_t i = 0; i < sz; i++)
        {
          tmp[i] = _start[i];
        }
        //释放旧空间
        delete[] _start;
      }
      //指向新空间
      _start = tmp;
      _finish = _start + sz;
      _end_of_storage = _start + n;
    }
  }


关于reserve中的深浅拷贝问题


在reserve的实现中,我们不能使用memcpy来拷贝数据,否则会发生浅拷贝的问题,导致在析构报错。对于简单的内置类型或自定义类型,memcpy是一个不错的选择,既高效又实用。但是一旦元素涉及到资源申请,memcpy只是简单的将一个元素的值拷贝给另一个元素,并不会将该元素指向的空间的内容全部拷贝给另一个元素。所以,我们必须手动实现深拷贝。

🍁错误示例

  void reserve(size_t n)
  {
    if (n > capacity())
    {
      size_t sz = size();
      //使用中间变量,防止new失败
      iterator tmp = new T[n];
      if (_start)
      {
        //浅拷贝
        memcpy(tmp, _start, sizeof(T)*size());
        delete[] _start;
      }
      //指向新空间
      _start = tmp;
      _finish = _start + sz;
      _end_of_storage = _start + n;
    }
  }


resize——扩容并初始化


  void resize(size_t n, T val = T())
  {
    //n<size(),删除数据
    if (n < size())
    {
      _finish = _start + n;
    }
    else
    {
      if (n > capacity)
        reserve(n); //扩容
      while (_finish != _start + n)
      {
        //初始化
        *_finish = val;
        ++_finish;
      }
    }
  }


push_back——尾插


  void push_back(const T& val)
  {
    //扩容
    if (_finish == _end_of_storage)
    {
      reserve(capacity() == 0 ? 4 : capacity() * 2);
    }
    //插入元素
    *_finish = val;
    ++_finish;
  }


pop_back——尾删


  void pop_back()
  {
    assert(!empty());
    --_finish;
  }


insert——插入


  iterator insert(iterator pos,T val=T()) //使用匿名构造
  {
    assert(pos >= _start);
    assert(pos <= _finish);
    //扩容
    if (_finish == _end_of_storage)
    {
      size_t len = pos - _start;
      reserve(capacity() == 0 ? 4 : capacity() * 2);
      //扩容后更新pos的位置,解决pos失效的问题
      pos =_start + len;
    }
    //挪动数据
    iterator end = _finish-1;
    while (end >= pos)
    {
      *(end + 1) = *end;
      --end;
    }
    //插入元素
    *pos = val;
    ++_finish;
    return pos;
  }


erase——删除


  iterator erase(iterator pos)
  {
    assert(pos >= _start);
    assert(pos < _finish);
    //挪动数据
    iterator end = pos + 1;
    while (end != _finish)
    {
      *(end - 1) = *end;
      ++end;
    }
    --_finish;
    return pos;
  }


empty——判空


  bool empty()
  {
    return size() == 0;
  }


[]重载——访问元素


  T& operator[](size_t pos)
  {
    assert(pos < size());
    return _start[pos];
  }
  const T& operator[](size_t pos) const
  {
    assert(pos < size());
    return _start[pos];
  }


传值构造


  vector<int> v1(10, 5); //用10个5来构造
  vector(size_t n, const T& val = T())
    :_start(nullptr),
    _finish(nullptr),
    _end_of_storage(nullptr)
  {
    //扩容
    reserve(n);
    //尾插
    for (size_t i = 0; i < n; i++)
    {
      push_back(val);
    }
  }
  //重载版本,防止调用时与迭代器区间构造混淆
  vector(int n, const T& val = T())
    :_start(nullptr),
    _finish(nullptr),
    _end_of_storage(nullptr)
  {
    reserve(n);
    for (size_t i = 0; i < n; i++)
    {
      push_back(val);
    }
  }



迭代器区间构造


  vector<int> v1(10, 5); 
  vector<string> v2(v1.begin(), v1.end()); //迭代器区间构造
  template<class InputIterator>
  vector(InputIterator begin, InputIterator end)
  {
    //扩容
    reserve(end - begin);
    while (begin != end)
    {
      push_back(*begin);
      ++begin;
    }
  }


赋值重载

  void swap(vector<T>& v)
  {
    std::swap(_start, v._start);
    std::swap(_finish, v._finish);
    std::swap(_end_of_storage, v._end_of_storage);
  }
  vector<T>& operator=(vector<T> v)
  {
    swap(v);
    return*this;
  }



拷贝构造


  vector(const vector<T>& v)
  {
    //写法1
    //_start = new T[v.capacity()];
    //for (size_t i = 0; i < v.size(); i++)
    //{
    //  _start[i] = v._start[i];
    //}
    //_finish = _start + v.size();
    //_end_of_storage = _start + v.capacity();
    //写法2
    vector<T> tmp(v.begin(), v.end());
    swap(tmp);
  }


拷贝构造中的深浅拷贝问题


在写法1中,如果使用memcpy,同样会发生浅拷贝的问题。

🍁错误示例

    vector(const vector<T>& v)
    {
      _start = new T[v.capacity()];
      //浅拷贝
      memcpy(_start, v._start, sizeof(T)*v.size());
      _finish = _start + v.size();
      _end_of_storage = _start + v.capacity();
    }


析构函数


  ~vector()
  {
    delete[] _start;
    _start = _finish = _end_of_storage = nullptr;
  }


完整源码


#include<iostream>
#include<assert.h>
using namespace std;
namespace hxy
{
  template<class T>
  class vector
  {
  public:
    typedef T* iterator;
    typedef const T* const_iterator;
    vector()
      :_start(nullptr),
      _finish(nullptr),
      _end_of_storage(nullptr)
    {}
    vector(size_t n, const T& val = T())
      :_start(nullptr),
      _finish(nullptr),
      _end_of_storage(nullptr)
    {
      //扩容
      reserve(n);
      //尾插
      for (size_t i = 0; i < n; i++)
      {
        push_back(val);
      }
    }
    vector(int n, const T& val = T())
      :_start(nullptr),
      _finish(nullptr),
      _end_of_storage(nullptr)
    {
      reserve(n);
      for (size_t i = 0; i < n; i++)
      {
        push_back(val);
      }
    }
    void swap(vector<T>& v)
    {
      std::swap(_start, v._start);
      std::swap(_finish, v._finish);
      std::swap(_end_of_storage, v._end_of_storage);
    }
    vector<T>& operator=(vector<T> v)
    {
      swap(v);
      return*this;
    }
    vector(const vector<T>& v)
    {
      //写法1
      _start = new T[v.capacity()];
      for (size_t i = 0; i < v.size(); i++)
      {
        _start[i] = v._start[i];
      }
      _finish = _start + v.size();
      _end_of_storage = _start + v.capacity();
      写法2
      //vector<T> tmp(v.begin(), v.end());
      //swap(tmp);
    }
    template<class InputIterator>
    vector(InputIterator begin, InputIterator end)
    {
      reserve(end - begin);
      while (begin != end)
      {
        push_back(*begin);
        ++begin;
      }
    }
    ~vector()
    {
      delete[] _start;
      _start = _finish = _end_of_storage = nullptr;
    }
    iterator begin()
    {
      return _start;
    }
    iterator end()
    {
      return _finish;
    }
    const_iterator begin() const
    {
      return _start;
    }
    const_iterator end() const
    {
      return _finish;
    }
    size_t size() const
    {
      return _finish - _start;
    }
    size_t capacity() const
    {
      return _end_of_storage - _start;
    }
    void reserve(size_t n)
    {
      if (n > capacity())
      {
        size_t sz = size();
        //使用中间变量,防止new失败
        iterator tmp = new T[n];
        if (_start)
        {
          //memcpy(tmp, _start, sizeof(T) * size());
          //转移数据
          for (size_t i = 0; i < sz; i++)
          {
            tmp[i] = _start[i];
          }
          //释放旧空间
          delete[] _start;
        }
        //指向新空间
        _start = tmp;
        _finish = _start + sz;
        _end_of_storage = _start + n;
      }
    }
    void resize(size_t n, T val = T())
    {
      //n<size(),删除数据
      if (n < size())
      {
        _finish = _start + n;
      }
      else
      {
        if (n > capacity)
          reserve(n); //扩容
        while (_finish != _start + n)
        {
          //初始化
          *_finish = val;
          ++_finish;
        }
      }
    }
    void push_back(const T& val)
    {
      //扩容
      if (_finish == _end_of_storage)
      {
        reserve(capacity() == 0 ? 4 : capacity() * 2);
      }
      //插入元素
      *_finish = val;
      ++_finish;
    }
    void pop_back()
    {
      assert(!empty());
      --_finish;
    }
    iterator insert(iterator pos,T val=T()) //使用匿名构造
    {
      assert(pos >= _start);
      assert(pos <= _finish);
      //扩容
      if (_finish == _end_of_storage)
      {
        size_t len = pos - _start;
        reserve(capacity() == 0 ? 4 : capacity() * 2);
        //扩容后更新pos的位置,解决pos失效的问题
        pos =_start + len;
      }
      //挪动数据
      iterator end = _finish-1;
      while (end >= pos)
      {
        *(end + 1) = *end;
        --end;
      }
      //插入元素
      *pos = val;
      ++_finish;
      return pos;
    }
    iterator erase(iterator pos)
    {
      assert(pos >= _start);
      assert(pos < _finish);
      //挪动数据
      iterator end = pos + 1;
      while (end != _finish)
      {
        *(end - 1) = *end;
        ++end;
      }
      --_finish;
      return pos;
    }
    T& operator[](size_t pos)
    {
      assert(pos < size());
      return _start[pos];
    }
    const T& operator[](size_t pos) const
    {
      assert(pos < size());
      return _start[pos];
    }
    bool empty()
    {
      return size() == 0;
    }
  private:
    iterator _start;
    iterator _finish;
    iterator _end_of_storage;
  };
}
目录
相关文章
|
4天前
|
编译器 C语言 C++
【c++丨STL】list模拟实现(附源码)
本文介绍了如何模拟实现C++中的`list`容器。`list`底层采用双向带头循环链表结构,相较于`vector`和`string`更为复杂。文章首先回顾了`list`的基本结构和常用接口,然后详细讲解了节点、迭代器及容器的实现过程。 最终,通过这些步骤,我们成功模拟实现了`list`容器的功能。文章最后提供了完整的代码实现,并简要总结了实现过程中的关键点。 如果你对双向链表或`list`的底层实现感兴趣,建议先掌握相关基础知识后再阅读本文,以便更好地理解内容。
14 1
|
17天前
|
算法 C语言 C++
【c++丨STL】list的使用
本文介绍了STL容器`list`的使用方法及其主要功能。`list`是一种双向链表结构,适用于频繁的插入和删除操作。文章详细讲解了`list`的构造函数、析构函数、赋值重载、迭代器、容量接口、元素访问接口、增删查改操作以及一些特有的操作接口如`splice`、`remove_if`、`unique`、`merge`、`sort`和`reverse`。通过示例代码,读者可以更好地理解如何使用这些接口。最后,作者总结了`list`的特点和适用场景,并预告了后续关于`list`模拟实现的文章。
33 7
|
2月前
|
存储 编译器 C语言
【c++丨STL】vector的使用
本文介绍了C++ STL中的`vector`容器,包括其基本概念、主要接口及其使用方法。`vector`是一种动态数组,能够根据需要自动调整大小,提供了丰富的操作接口,如增删查改等。文章详细解释了`vector`的构造函数、赋值运算符、容量接口、迭代器接口、元素访问接口以及一些常用的增删操作函数。最后,还展示了如何使用`vector`创建字符串数组,体现了`vector`在实际编程中的灵活性和实用性。
64 4
|
2月前
|
C语言 C++ 容器
【c++丨STL】string模拟实现(附源码)
本文详细介绍了如何模拟实现C++ STL中的`string`类,包括其构造函数、拷贝构造、赋值重载、析构函数等基本功能,以及字符串的插入、删除、查找、比较等操作。文章还展示了如何实现输入输出流操作符,使自定义的`string`类能够方便地与`cin`和`cout`配合使用。通过这些实现,读者不仅能加深对`string`类的理解,还能提升对C++编程技巧的掌握。
77 5
|
2月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
60 2
|
16天前
|
存储 对象存储 C++
C++ 中 std::array<int, array_size> 与 std::vector<int> 的深入对比
本文深入对比了 C++ 标准库中的 `std::array` 和 `std::vector`,从内存管理、性能、功能特性、使用场景等方面详细分析了两者的差异。`std::array` 适合固定大小的数据和高性能需求,而 `std::vector` 则提供了动态调整大小的灵活性,适用于数据量不确定或需要频繁操作的场景。选择合适的容器可以提高代码的效率和可靠性。
39 0
|
2月前
|
存储 算法 Linux
【c++】STL简介
本文介绍了C++标准模板库(STL)的基本概念、组成部分及学习方法,强调了STL在提高编程效率和代码复用性方面的重要性。文章详细解析了STL的六大组件:容器、算法、迭代器、仿函数、配接器和空间配置器,并提出了学习STL的三个层次,旨在帮助读者深入理解和掌握STL。
55 0
|
20天前
|
存储 编译器 C语言
【c++丨STL】vector模拟实现
本文深入探讨了 `vector` 的底层实现原理,并尝试模拟实现其结构及常用接口。首先介绍了 `vector` 的底层是动态顺序表,使用三个迭代器(指针)来维护数组,分别为 `start`、`finish` 和 `end_of_storage`。接着详细讲解了如何实现 `vector` 的各种构造函数、析构函数、容量接口、迭代器接口、插入和删除操作等。最后提供了完整的模拟实现代码,帮助读者更好地理解和掌握 `vector` 的实现细节。
30 0
|
3月前
|
存储 程序员 C++
C++常用基础知识—STL库(2)
C++常用基础知识—STL库(2)
89 5
|
3月前
|
存储 C++ 索引
【C++打怪之路Lv9】-- vector
【C++打怪之路Lv9】-- vector
27 1