首先由于自定义类不能和已有类重名,所以在自定义命名空间中进行vector类的模拟
namespace aiyimu
{
template<class T>//类模板
class vector
{
public:
//迭代器
typedef T* iterator;
typedef const T* const_iterator;
//成员函数
private:
//成员变量
iterator _start;//起始位置
iterator _finish;//所含元素末尾
iterator _endofstorage;//开辟的空间末尾位置
};
}
无参构造函数
无参构造进行成员函数的初始化即可
vector()
:_start(nullptr)
,_finish(nullptr)
,_endofstorage(nullptr)
{}
迭代器构造函数
push_back()
(尾插函数)后面会进行实现template <class InputIterator>//定义一个模板参数
vector(InputIterator first, InputIterator last)
:_start(nullptr)
,_finish(nullptr)
,_endofstorage(nullptr)
{
//从first到last构造
while (first != last)
{
push_back(*first);
++first;
}
}
传统写法*
v.capacity()
的连续内存空间,将_start指向该空间起始位置,起始位置设置完成,同理更新_finish, _endofstorage (capacity()是返回vector空间大小的函数,后面实现)vector(const vector<T>& v)
{
_start = new T[v.capacity()];
_finish = _start + v.size();
_endofstorage = _start + capacity();
//memcpy(_start, v._start, v.size() * sizeof(T));
//同reserve()一样,memcpy是浅拷贝,会出现问题
for (size_t i = 0; i < sz; i++)
{
tmp[i] = _start[i];
}
}
现代写法*
swap()
函数进行拷贝操作(将tmp的元素给*this)void swap(vector<T>& v)
{
std::swap(_start, v._start);
std::swap(_finish, v._finish);
std::swap(_endofstorage, v._endofstorage);
}
vector(const vector<T>& v)
:_start(nullptr)
,_finish(nullptr)
,_endofstorage(nullptr)
{
vector<T> tmp(v.begin(), v.end());;//利用构造函数直接将v构造到临时vector中
swap(tmp);//和this->swap(tmp)相同
}
为什么不能使用memcpy()?
如果拷贝的是内置类型的元素,memcpy即高效又不会出错;
但如果拷贝的是自定义类型元素,并且自定义类型元素中涉及到资源管理时,就会出错,因为memcpy的拷贝实际是浅拷贝。
具体过程如下图:
当使用memcpy()
拷贝后,_start指向v._start所指向的空间,拷贝完后释放原空间,此时_start就变为了野指针。
~vector()
{
//如果_start不为空,释放空间并置空指针
if (_start)
{
delete[] _start;
_start = _finish = _endofstorage = nullptr;
}
}
swap函数
直接交换即完成赋值vector<T>& operator=(vector<T> v)
{
swap(v);
return *this;
}
const T& operator[](size_t pos) const
{
assert(pos < size());//防止越界
return _start[pos];
}
T& operator[](size_t pos)
{
assert(pos < size());//防止越界
return _start[pos];
}
首先我们在类进行了如下操作:分别typedef了两种迭代器(const和非const)
public:
typedef T* iterator;
typedef const T* const_iterator;
const iterator begin() const
{
return _start;
}
const iterator end() const
{
return _finish;
}
iterator begin()
{
return _start;
}
iterator end()
{
return _finish;
}
size()
和capacity()
前面已经进行了使用,是为了访问vector的 大小 和 容量 的//返回此时vector的大小
size_t size()
{
return _finish - _start;
}
//返回此时vector的容量
size_t capacity()
{
return _endofstorage - _start;
}
const size_t size() const
{
return _finish - _start;
}
const size_t capacity() const
{
return _endofstorage - _start;
}
法一
if(_start)
)void push_back(const t& x)
{
//判断扩容
if (_finish == _endofstorage)
{
size_t newcapacity = capacity() == 0 ? 4 : capacity() * 2;//扩容
t* tmp = new t[newcapacity];
if (_start)//当_start不为空进来(第一次增容不用进)
{
memcpy(tmp, _start, sizeof(t) * size());//将原本存储的数据拷贝到tmp中
delete[] _start;
}
_finish = _start + size();
_start = tmp;//这两行顺序不能变:先赋值_start会改变size()的返回值
_endofstorage = _start + newcapacity;
}
*_finish = x;
++_finish;
}
法二
void push_back(const T& x)
{
//判断扩容
if (_finish == _endofstorage)
{
size_t newCapacity = capacity() == 0 ? 4 : capacity() * 2;//扩容
T* tmp = new T[newCapacity];
size_t sz = size();
if (_start)//当_start不为空进来(第一次增容不用进)
{
memcpy(tmp, _start, sizeof(T) * size());//将原本存储的数据拷贝到tmp中
delete[] _start;
}
_start = tmp;
_finish = _start + sz;
_endofstorage = _start + newCapacity;
}
*_finish = x;
++_finish;
}
法三
reserve()
(扩容函数)后,直接调用reserve()
,再进行插入即可reserve()
下面实现void push_back(const T& x)
{
//判断扩容
if (_finish == _endofstorage)
{
reserve(capacity() == 0 ? 4 : capacity() * 2);
}
*_finish = x;
++_finish;
}
reserve()
函数只在 vector 对象中没有足够的内存空间时才会进行内存分配操作,以便预留一定的内存空间,避免频繁的内存分配操作影响程序性能。if(_start)
)void reserve(size_t n)
{
//当_finish == _endofstorage时,此时没有多余内存空间,进行增容
if (_finish == _endofstorage)
{
T* tmp = new T[n];
size_t sz = size();
if (_start)//当_start不为空进来(第一次增容不用进)
{
//memcpy是浅拷贝,在一些使用场景中会出现问题
//memcpy(tmp, _start, sizeof(T) * size());//将原本存储的数据拷贝到tmp中
for (size_t i = 0; i < sz; i++)
{
//T是int,一个个拷贝没有问题
//T是string,每次调用=也是深拷贝
tmp[i] = _start[i];
}
delete[] _start;
}
_start = tmp;
_finish = _start + sz;
_endofstorage = _start + n;
}
}
resize()
允许增容+缩容,当 (n<当前大小)时进行缩容,反之扩容resize()
此时为扩容+初始化)void resize(size_t n, const T& val = T())
{
//缩小size
if (n < size())
{
_finish = _start + n;
}
//扩大
else
{
//判断增容
if (n > capacity())
{
reserve(n);
}
while (_finish != _start + n)//将多出的空间赋值val
{
*_finish = val;
++_finish;
}
}
}
insert()
- 向任意位置插入字符insert()
需要返回一个指向插入的元素的迭代器。最后返回迭代器pos。iterator insert(iterator pos, const T& x)
{
//防止pos越界
assert(pos >= _start);
assert(pos <= _finish);
//判读扩容
if (_finish == _endofstorage)
{
//扩容后pos会失效,所以在扩容后更新pos
size_t len = pos - _start;
reserve(capacity() == 0 ? 4 : capacity() * 2);
//扩容后更新pos
pos = _start + len;
}
iterator end = _finish - 1;
//将插入位置后面的元素后移
while (end >= pos)
{
*(end + 1) = *end;
--end;
}
*pos = x;//插入元素
++_finish;
return pos;
}
iterator erase(iterator pos)
{
assert(pos >= _start);
assert(pos <= _finish);
iterator begin = pos + 1;
//将pos位后面的元素前移
while (begin < _finish)
{
*(begin - 1) = *(begin);
++begin;
}
--_finish;
return pos;
}
vector的模拟实现