深入了解vector

1. vector的介绍及使用

1.1 vector的介绍

在C++中,vector是一个模板类,它是一个多功能的、能够操作多种数据结构和算法的函数库。它是C++标准模板库中的部分内容,用于表示动态数组。vector可以自动扩展以容纳新元素,并且可以在任何位置插入或删除元素。

vector的优点:

  • vector是一个动态数组,可以自动扩展以容纳新元素。
  • vector支持随机访问,可以通过下标访问元素。
  • vector支持在任何位置插入或删除元素。

vector的缺点:

  • vector的插入和删除操作可能会导致内存重新分配,从而导致性能下降。
  • vector只能存储相同类型的元素。

1.2 vector的使用

1.2.1 vector的定义((constructor)构造函数声明)

default (1) vector (); //无参构造
fill (2) explicit vector (size_type n, const value_type& val = value_type()) // 构造并初始化n个val
range (3) vector (InputIterator first, InputIterator last); //使用迭代器进行初始化构造
copy (4) vector (const vector& x); //拷贝构造

1.2.2 vector iterator 的使用

在这里插入图片描述

vector的迭代器是一种指针,它可以遍历vector中的元素。vector的迭代器支持随机访问,可以通过下标访问元素。反向迭代器是一种特殊的迭代器,它可以从后向前遍历容器中的元素。
反向迭代器和正向迭代器的区别在于:对正向迭代器进行++操作时,迭代器会指向容器中的后一个元素;而对反向迭代器进行++操作时,迭代器会指向容器中的前一个元素。

const_iterator是一种只读迭代器,它不能修改vector中的元素。只读迭代器可以用于遍历vector中的元素,但不能用于修改vector中的元素。

#include <iostream>
#include <vector>
using namespace std;

int main()
{
    vector<int> v = { 1, 2, 3, 4, 5 };

    // 使用begin和end遍历vector中的元素
    for (auto i = v.begin(); i != v.end(); ++i)
        cout << *i << " ";
    cout << endl;

    // 使用rbegin和rend遍历vector中的元素
    for (auto i = v.rbegin(); i != v.rend(); ++i)
        cout << *i << " ";
    cout << endl;

    return 0;
}

在这里插入图片描述

1.2.3 vector Capacity

在这里插入图片描述
用如下代码查看vector 空间增长问题
VS平台下:

#include <iostream>
#include <vector>
using namespace std;

int main()
{
	vector<int> v;
	cout << "capacity: " << v.capacity() << endl;
	while (1)
	{
		v.push_back(1);
		if (v.size() == v.capacity())
		{
			cout << "capacity: " << v.capacity() << endl;
		}
	}

	return 0;
}

在这里插入图片描述
gcc平台下:
在这里插入图片描述
capacity的代码在vs和g++下分别运行会发现,vs下capacity是按1.5倍增长的,g++是按2倍增长的。

1.2.4 vector Modifiers

在这里插入图片描述
assign()是vector的成员函数之一,它可以用于将新元素分配给vector。assign()函数有多个重载版本,可以接受不同类型的参数。下面是一些常见的用法:

// assign()函数的第一个版本
vector<int> v1;
v1.assign(5, 10); // 将5个值为10的元素分配给v1

// assign()函数的第二个版本
vector<int> v2;
int arr[] = {1, 2, 3, 4, 5};
v2.assign(arr, arr + 5); // 将数组arr中的元素分配给v2

// assign()函数的第三个版本
vector<int> v3;
vector<int> v4;
v3.assign(v4.begin(), v4.end()); // 将v4中的元素分配给v3

emplace()是vector的成员函数之一,它可以用于在vector中构造新元素。emplace()函数有多个重载版本,可以接受不同类型的参数。下面是一些常见的用法:

// emplace()函数的第一个版本
vector<pair<int, string>> v1;
v1.emplace_back(1, "hello"); // 在v1中构造一个pair<int, string>类型的元素

// emplace()函数的第二个版本
vector<complex<double>> v2;
v2.emplace(v2.begin(), 1.0, 2.0); // 在v2的开头构造一个complex<double>类型的元素

// emplace()函数的第三个版本
vector<string> v3;
v3.emplace_back("hello"); // 在v3中构造一个string类型的元素在这里插入代码片
1.2.4 vector 迭代器失效问题

vector的迭代器失效问题是指,当vector中的元素被添加、删除或插入时,迭代器可能会失效。如果一个迭代器失效了,那么它就不能再用于访问vector中的元素。

下面是一些常见的导致迭代器失效的操作:

  1. 会引起其底层空间改变的操作,都有可能是迭代器失效,比如:resize、reserve、insert、assign、
    push_back等。
  2. 指定位置元素的删除操作–erase
#include <iostream>
using namespace std;
#include <vector>

int main()
{
 int a[] = { 1, 2, 3, 4 };
 vector<int> v(a, a + sizeof(a) / sizeof(int));
 // 使用find查找3所在位置的iterator
 vector<int>::iterator pos = find(v.begin(), v.end(), 3);
 // 删除pos位置的数据,导致pos迭代器失效。
 v.erase(pos);
 cout << *pos << endl; // 此处会导致非法访问
 return 0;
}

在这里插入图片描述

erase删除pos位置元素后,pos位置之后的元素会往前搬移,没有导致底层空间的改变,理论上讲迭代器不应该会失效,但是:如果pos刚好是最后一个元素,删完之后pos刚好是end的位置,而end位置是没有元素的,那么pos就失效了。因此删除vector中任意位置上元素时,vs就认为该位置迭代器失效
了。
迭代器失效解决办法:在使用前,对迭代器重新赋值即可。

#include <iostream>
using namespace std;
#include <vector>
int main()
{
 int a[] = { 1, 2, 3, 4 };
 vector<int> v(a, a + sizeof(a) / sizeof(int));
 // 使用find查找3所在位置的iterator
 vector<int>::iterator pos = find(v.begin(), v.end(), 3);
 // 删除pos位置的数据,导致pos迭代器失效。
 pos = v.erase(pos); 
 // 按照下面方式写,运行时程序会崩溃,因为erase(pos)之后
 // pos位置的迭代器就失效了
 // s.erase(pos);
 cout << *pos << endl; // 此处会导致非法访问
 return 0;
}

在这里插入图片描述

2. vector模拟实现

在这里插入图片描述
在stl源码中,可以看到vector使用三个iterator来实现的。

template<class T>
	class vector
	{
	public:
		typedef T* iterator;
		typedef const T* const_iterator;
		vector()
		{}

		//vector<int> v(10, 5);
		vector(size_t n, const T& val = T())
		{
			reserve(n);
			for (size_t i = 0; i < n; ++i)
			{
				push_back(val);
			}
		}

		vector(int n, const T& val = T())
		{
			reserve(n);
			for (int i = 0; i < n; ++i)
			{
				push_back(val);
			}
		}

		vector(const vector<T>& v)
		{
			_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();
		}

		// [first, last)
		template <class InputIterator>
		vector(InputIterator first, InputIterator last)
		{
			while (first != last)
			{
				push_back(*first);
				++first;
			}
		}

		~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;
		}

		void resize(size_t n, T val = T())
		{
			if (n < size())
			{
				// 删除数据
				_finish = _start + n;
			}
			else
			{
				if (n > capacity())
					reserve(n);

				while (_finish != _start + n)
				{
					*_finish = val;
					++_finish;
				}
			}
		}

		void reserve(size_t n)
		{
			if (n > capacity())
			{
				size_t sz = size();
				T* 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;
			}
		}

		void push_back(const T& x)
		{
			if (_finish == _end_of_storage)
			{
				reserve(capacity() == 0 ? 4 : capacity() * 2);
			}

			*_finish = x;
			++_finish;
		}

		void pop_back()
		{
			assert(!empty());

			--_finish;
		}

		iterator insert(iterator pos, const T& val)
		{
			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 start = pos + 1;
			while (start != _finish)
			{
				*(start - 1) = *start;
				++start;
			}

			--_finish;

			return pos;
		}

		size_t capacity() const
		{
			return _end_of_storage - _start;
		}

		size_t size() const
		{
			return _finish - _start;
		}

		bool empty()
		{
			return _start == _finish;
		}

		T& operator[](size_t pos)
		{
			assert(pos < size());

			return _start[pos];
		}

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

			return _start[pos];
		}

	private:
		iterator _start = nullptr;
		iterator _finish = nullptr;
		iterator _end_of_storage = nullptr;
	};
  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ly@눈_눈

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值