【C++】—— vector 的模拟实现

news2025/1/17 0:51:40

【C++】—— vector 的模拟实现

  • 0 前言
  • 1 vector 的成员变量
    • 1.1 stl 库中的 vector 成员变量
    • 1.2 模拟实现 vector 成员变量
  • 2 迭代器
  • 3 size、capacity、empty
  • 4 opreator[ ]
  • 5 reserve
    • 5.1 初版 reserve
    • 5.2 _finish 的处理
    • 5.3 深拷贝
    • 5.4 终版
  • 6 push_back 与 pop_back
  • 7 打印函数
    • 7.1 初写打印
    • 7.2 typename 关键字
    • 7.3 打印函数进阶
  • 8 insert
    • 8.1 insert 初版
    • 8.2 迭代器失效
      • 8.2.1 情况一:类似野指针
      • 8.2.2 algorithm 库中的查找函数
      • 8.2.3 情况二:insert 后再对 pos 进行访问
        • 8.2.3.1 不扩容的情况
        • 8.2.3.1 扩容的情况
      • 8.2.4、解决方法
        • 8.2.4.1、想法一
        • 8.2.4.2、想法二
    • 8.3 总结
  • 9 erase
  • 10 resize
  • 11 构造函数系列
    • 11.1 默认构造
    • 11.2 拷贝构造
    • 11.3 迭代器区间构造 与 n 个 value 构造
      • 11.3.1 迭代器区间构造
      • 11.3.2 n 个value构造
      • 11.3.3 注意事项
  • 12 赋值运算符重载
    • 12.1 传统写法
    • 12.2 现代写法
  • 13 源码

0 前言

  前面,我们学习了 s t r i n g string string,并模拟实现了它,下面我们来学习 v e c t o r vector vector
   v e c t o r vector vector 的底层即我们前面曾经学过的顺序表,它属于 STL库,为了让我们更加深入理解 v e c t o r vector vector,接下来我们将模拟实现 v e c t o r vector vector
    
  

1 vector 的成员变量

1.1 stl 库中的 vector 成员变量

  我们学习数据结构时,曾经模拟实现过顺序表。当时我们用一个指针 a a a 管理从堆开辟的空间, s i z e size size 表示当前有效数据的个数, c a p a c i t y capacity capacity 表示当前空间的大小。
  
  但是, s t l stl stl 中的 v e c t o r vector vector 的成员变量有所差别:它的成员变量是三个迭代器

在这里插入图片描述

  
  我们知道,迭代器全部都是 t y p e d e f typedef typedef 出来的,那 v e c t o r vector vector 的迭代器是什么呢?

在这里插入图片描述

  可以看到,其迭代器就是 原生指针

  那startfinishend_of_storage 变量又是什么意思呢?面对未知的东西,我们应大胆假设,小心求证

  不管他底层如何, v e c t o r vector vector 终究逃不过扩容,那我们不难猜想:start 肯定是一个指针指向一个开始finish指向某个结束,至于end_of_storage s t o r a g e storage storage 是存储的意思,所以我们大胆猜测end_of_storage与空间相关,可能就是空间结束位置;那这样的话finish 可能表示数据的结束

  那我们怎么取求证呢?我们可以通过一些我们熟悉功能的函数来确认

在这里插入图片描述

  
  可以看到, b e g i n begin begin 返回的是第一个有效数据位置,这里是start;而 e n d end end 返回的是最后一个有效数据的下一个位置,这里返回的是finish c a p a c i t y capacity capacity 表示空间的大小,这里返回的是end_of_storage - begin()
  因此我们的猜测是正确的,那么 v e c t o r vector vector 的底层结构如下:

在这里插入图片描述

finish 指向的是有效空间的下一个位置
  
  

1.2 模拟实现 vector 成员变量

  我们模拟实现的 v e c t o r vector vector 的成员变量,也学习库中的形式

namespace my_vector
{
	template<class T>
	class vector
	{
	public:
		//···

	private:
		iterator _start = nullptr;
		iterator _finish = nullptr;
		iterator _end_of_storage = nullptr;
	};
}

  这里,为了与库中的 v e c t o r vector vector 区别开,我们使用命名空间

  同时,因为顺序表应满足不同数据类型的存储,我们实现的不是某个具体类型的类,而是类模板。既然是模板,那么函数的声明与定义就不能放在不同的文件中,因此声明和定义全在vector.h文件中,不再需要vector.cpp文件。

  
  

2 迭代器

  因为 v e c t o r vector vector 的迭代器是一个原生指针,与 s t r i n g string string 类似,很简单,我们直接看代码

typedef T* iterator;
typedef const T* const_iterator;

iterator begin()
{
	return _start;
}

iterator end()
{
	return _finish;
}

const_iterator begin() const
{
	return _start;
}

const_iterator end() const
{
	return _finish;
}

  因为 _ f i n i s h finish finish 本就就是只想最后一个有效数据的下一位,因此我们end()我们直接返回 _finish 即可

  
  

3 size、capacity、empty

  这三个函数都很简单,我们直接看代码

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

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

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

  
  

4 opreator[ ]

   v e c t o r vector vector 中的 o p r e a t o r opreator opreator[] 与 s t r i n g string string 中的 o p e r a t o r operator operator[] 类似,都是访问第 i i i 个数据的元素。同样, v e c t o r vector vector o p e r a t o r operator operator[] 也要用 a s s e r t assert assert 断言来判断下标是否合法

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

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

  
  这里顺便提一下, v e c t o r vector vector 还提供了 at 函数 接口。
   a t at at 函数与 o p e r a t o operato operato[] 功能相同,但他们检查越界的方式不同
   a t at at 函数是抛异常,而 o p e r a t o r operator operator[] 是断言
  
  

5 reserve

5.1 初版 reserve

void reserve(size_t n)
{
	if (n > capacity())
	{			
		T* tmp = new T[n];
		memcpy(tmp, _start, sizeof(T) * size());
		delete[] _start;
		_start = tmp;

		_finish = _start + size();
		_end_of_storage = _start + n;
	}
}

  在 s t r i n g string string 中,我们已经模拟实现 r e s e r v e reserve reserve 了, v e c t o r vector vector 中也是需要我们手动释放和拷贝的数据的,代码细节就不再细讲了。

  那上面代码有问题吗?
  有的

5.2 _finish 的处理

  首先是对 _ f i n i s h finish finish 的处理上
  这里我们更新_finish的方式是_start + size(),可是 size() 是怎么计算出来的?是通过_finish - _start计算出的,可是此时 _ s t a r t start start已经更新了,而_ f i n i s h finish finish还未更新 ,此时他们相减会出大问题。

在这里插入图片描述

  怎么解决呢?也很好办,那就是提前记录 s i z e size size() 的值

void reserve(size_t n)
{
	if (n > capacity())
	{
		//记下size()的值
		size_t old_size = size();

		T* tmp = new T[n];
		memcpy(tmp, _start, sizeof(T) * size());
		delete[] _start;
		_start = tmp;

		_finish = _start + old_size;
		_end_of_storage = _start + n;
	}
}

  
  

5.3 深拷贝

  还有一点,上述代码用 m e m c p y memcpy memcpy 拷贝是浅拷贝,而我们需要的是深拷贝

  用 m e m c p y memcpy memcpy 拷贝有什么问题呢?

void test7()
{
	vector<string> v1;

	v1.push_back("11111");
	v1.push_back("11111");
	v1.push_back("11111");
	v1.push_back("11111");
	print_container(v1);

	v1.push_back("11111");
	print_container(v1);
}

运行结果:

在这里插入图片描述

  程序崩溃了!
  为什么呢?问题出现在了扩容

  我们知道 v e c t o r vector vector< s t r i n g string string> 中,每个 s t r i n g string string 都指向一块堆空间。
  虽然 v e c t o r vector vector 是深拷贝,但是 v e c t o r vector vector 中的每个 s t r i n g string string 都是浅拷贝,而每个string都指向一块资源。这意味着新开辟的 t m p tmp tmp 中,每个 string 都是与对应旧空间的 string 指向同一块空间 d e l e t e delete delete 后,会调用析构函数,将旧空间指向的资源全部释放掉,此时新空间的每个 s t r i n g string string 都是指向一块已经被释放的空间,自然会出现问题。

在这里插入图片描述

  
  那如何解决呢?那就是对vector中的每个成员都进行深拷贝。我们不能再使用 m e m c p y memcpy memcpy,而是要自己赋值

void reserve(size_t n)
{
	if (n > capacity())
	{		
		size_t old_size = size();
	
		T* tmp = new T[n];
		for (size_t i = 0; i < old_size; ++i)
		{
			_tmp[i] = _start[i];
		}
		delete[] _start;
		_start = tmp;
	
		_finish = _start + old_size;
		_end_of_storage = _start + n;
	}
}

  _tmp[i] = _start[i];是赋值,本质是调用自定义类型的赋值重载函数,实现了每个成员的深拷贝,而[ ],则是前面实现的operator[]重载函数

在这里插入图片描述

  
  

5.4 终版

void reserve(size_t n)
{
	if (n > capacity())
	{		
		size_t old_size = size();
	
		T* tmp = new T[n];
		for (size_t i = 0; i < old_size; ++i)
		{
			_tmp[i] = _start[i];
		}
		delete[] _start;
		_start = tmp;
	
		_finish = _start + old_size;
		_end_of_storage = _start + n;
	}
}

  
  

6 push_back 与 pop_back

   p u s h push push_ b a c k back back 直接往尾部放数据即可,但要注意空间不够要扩容

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

	*_finish = x;
	++_finish;
}

  
   p o p pop pop_ b a c k back back 也很简单,但要注意判断是否为空

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

  
  

7 打印函数

  STL库 中, v e c t o r vector vector没有重载 o p e r a t o r operator operator<< 的
  为什么呢?因为库事先并不知道你要以什么形式打印:是连续打印,还是每打印一个数据就空格还是每打印一个就换行?而且自己实现一个打印函数并不困难。因此库就交给程序员自己实现。
  

7.1 初写打印

  那么我们就自己实现一个打印函数吧。
  打印函数是一个全局函数

template<class T>
void print_vector(const vector<T>& v)
{
	vector<T>::const_iterator it = v.begin();
	while (it != v.end())
	{
		cout << *it << " ";
		++it;
	}
	cout << endl;
}

  但上述函数写的是有问题的

在这里插入图片描述

  

7.2 typename 关键字

  为什么呢?
  问题出现在vector<T>::const_iterator it;

  可以直接突破类域,用类名::方式去类中取东西,能取到两种:一种是 静态成员变量,另一种是在 类中 t y p e d e f typedef typedef 的类型
  
  C++规定:类模板没有被实例化之前,编译器是不会往里面去取东西的
  编译器走到 p r i n t print print_ v e c t o r vector vector 时, v e c t o r vector vector 模板还没被实例化出来。而因为不敢去模板中取,编译器不知道 c o n s t const const_ i t e r a t o r iterator iterator类型 还是 静态成员变量,如果取到静态成员变量,后面还跟着变量 it 就会出大问题。因此编译器报错。

  要解决上述问题,需在前面加上 t y p e n a m e typename typename 关键字。加上 t y p e n a m e typename typename即明确告诉编译器 c o n s t const const_ i t e r a t o r iterator iterator 是类型。(可以看做编译器的甩锅。编译器:你说了是类型的啦,我给你过了,如果他是成员变量的话那也是你的问题,不怪我)
  但如果是取静态成员变量前面什么都不用加,因为如果真是静态成员变量,你不会像vector<T>::const_iterator it这样用的,直接就vector<T>::const_iterator;这样
  

template<class T>
void print_vector(const vector<T>& v)
{
	//没有实例化的类模板里面取东西,编译器不能区分const_iterator是类型还是静态成员变量
	typename vector<T>::const_iterator it = v.begin();
	while (it != v.end())
	{
		cout << *it << " ";
		++it;
	}
	cout << endl;
}

  这样就可以啦
  
  当然,这还有一种更简单的方式,用 a u t o auto auto 就没有那么麻烦,因为 a u t o auto auto 直接是由v.begin()推导而来

template<class T>
void print_vector(const vector<T>& v)
{	
	auto it = v.begin();
	while (it != v.end())
	{
		cout << *it << " ";
		++it;
	}
	cout << endl;
}

  
  

7.3 打印函数进阶

  但上述打印函数,稍稍改进一下,就可以变成所有类型容器的打印函数,而不仅仅是 v e c t o r vector vector 类型的

template<class Container>
void print_container(const Container& v)
{
	auto it = v.begin();
	while (it != v.end())
	{
		cout << *it << " ";
		++it;
	}
	cout << endl;
}

  需要注意的是判断条件while (it != v.end())只能是 !=,有些小伙伴可能会写while (it < v.end())。对于 s t r i n g string string v e c t o vecto vector 来说是没问题,因为他们本来就是原生指针;但对其他容器就不行了,比例 l i s t list list 链表,它的 i t e r a t o r iterator iterator 是一个,不存在比较大小的概念。
  
  

8 insert

8.1 insert 初版

void insert(iterator pos, const T& x)
{
	assert(pos >= begin() && pos <= end());
	if (_finish == _end_of_storage)
	{
		size_t n = size() == 0 ? 4 : 2 * size();
		reserve(n);
	}

	iterator i = end();
	while (pos != i)
	{
		*i = *(i - 1);
		--i;
	}

	*pos = x;
	++_finish;
}

  上述代码其实是有问题
  我们一起来看看
  

8.2 迭代器失效

8.2.1 情况一:类似野指针

  上述代码其实是有问题的
  我们先来测试一个不需要扩容

void test1()
{
	vector<int> v1;
	v1.reserve(10);
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);
	v1.insert(v1.begin() + 1, 10);

	print_vector(v1);
}

运行结果:

在这里插入图片描述

  没有问题
  

  那再来一个需要扩容

void test1()
{
	vector<int> v1;
	v1.reserve(4);
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);
	v1.insert(v1.begin() + 1, 10);

	print_vector(v1);
}

运行结果:

在这里插入图片描述

  可以看到,报错了

  为什么呢?这是因为 p o s pos pos 迭代器失效

在这里插入图片描述

  我们结合图来看就很容易看懂了。扩容后,pos 依然指向旧空间,此时旧空间已经被释放了 p o s pos pos 迭代器就是一个野指针,同时while (pos != i)判断语句永远也不会成立,造成死循环。

  怎么办呢?我们应记录 pos 迭代器与 _start 的相对位置再将 pos 映射到新空间中

void insert(iterator pos, const T& x)
{
	assert(pos >= begin() && pos <= end());
	if (_finish == _end_of_storage)
	{
		size_t len = pos - _start;
		size_t n = size() == 0 ? 4 : 2 * size();
		reserve(n);
		pos = _start + len;
	}

	iterator i = end();
	while (pos != i)
	{
		*i = *(i - 1);
		--i;
	}

	*pos = x;
	++_finish;
}

  

8.2.2 algorithm 库中的查找函数

  这里,介绍 a l g o r i t h m algorithm algorithm中的查找函数 f i n d find find 函数模板适用于所有的容器,只需要传一个迭代器区间给它,就能帮你查找

在这里插入图片描述

  库中的 f i n d find find函数 是一个模板

在这里插入图片描述

  需要注意的是,传值要传左闭右开区间

在这里插入图片描述

  
  

8.2.3 情况二:insert 后再对 pos 进行访问

8.2.3.1 不扩容的情况

  我们知道, i n s e r t insert insert 函数的 p o s pos pos 形参是一个迭代器,如果我们想在指定数据之前插入数据,就可以用 find 模板找到相应数据的迭代器

  现在,我用 f i n d find find函数 找到 p p p 为 5 的位置,在之前插入数据 100,并将 5 ∗ * 10

void test2()
{
	vector<int> v1;
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);
	v1.push_back(5);
	v1.push_back(6);

	int x = 5;
	auto p = std::find(v1.begin(), v1.end(), x);
	if (p != v1.end())
	{
		v1.insert(p, 40);
		*(p) *= 10;
	}

	print_container(v1);
}

运行结果:

在这里插入图片描述

  为什么是100 * 10 呢?
  这是因为迭代器失效了,在 i n s e r t insert insert 以后,我们可以认为迭代器 p 就已经失效了。我们结合图来理解

在这里插入图片描述

  用 i n s e r t insert insert 插入数据后, p p p 指向的位置已经变了。你以为它指向的还是 5,但实际上他已经变了
  

8.2.3.1 扩容的情况

  上述情况还不是最恐怖的,最恐怖的是扩容的情况

void test3()
{
	vector<int> v1;
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);

	int x = 2;
	auto p = std::find(v1.begin(), v1.end(), x);
	if (p != v1.end())
	{
		v1.insert(p, 100);
		*(p) *= 10;
	}
	
	print_container(v1);
}

运行结果:

在这里插入图片描述

  现在连100 都不乘10了,为什么呢?还是迭代器失效的问题,我们来看图

在这里插入图片描述

   p o s pos pos 的位置还是指向旧空间,这时再对 p o s pos pos 访问,就是类似于野指针的访问。
  

8.2.4、解决方法

8.2.4.1、想法一

  那应该怎么解决上述问题呢?
  在解决完第一种情况的代码中,我们在函数内是对 p o s pos pos 进行了更新的

在这里插入图片描述

  
   但是形参的改变并不影响实参,那我们可不可以用引用呢?

void insert(iterator& pos, const T& x)

  这样, p o s pos pos 是引用,就能改变函数外的 p p p 了。
  
  但这样是不行的,因为会影响传参

vector<int> v;
v1.insert(v1.begin(), 10);

  像这样,我们传递第一个迭代器,但这样传参,v1.begin()的返回值中间会产生一个临时变量,实际传递的就是这个临时变量。临时变量是常性,必须用 c o n s t const const 引用,普通的引用是传不进去的。
  
  那如果我加 c o n s t const const 呢?

void insert(const iterator& pos, const T& x)

  加了 c o n s t const const,里面的 p o s pos pos 就无法改变了,所以还是不行
  
  

8.2.4.2、想法二

  我们来看一下库中的 i n s e r t insert insert 是怎么处理的

在这里插入图片描述

  库中给了一个返回值,返回的是更新之后的 p o s pos pos

iterator insert(iterator pos, const T& x)
{
	assert(pos >= begin() && pos <= end());
	
	if (_finish == _end_of_storage)
	{
		size_t len = pos - _start;
		size_t n = size() == 0 ? 4 : 2 * size();
		reserve(n);
		pos = _start + len;
	}

	iterator i = end();
	while (pos != i)
	{
		*i = *(i - 1);
		--i;
	}

	*pos = x;
	++_finish;
	return pos;
}

  所以如果真要 i n s e r t insert insert 后继续访问 p p p一定要更新 p p p 再访问

void test4()
{
	vector<int> v1;
	v1.push_back(1);
	v1.push_back(2);
	v1.push_back(3);
	v1.push_back(4);

	int x = 2;
	auto p = std::find(v1.begin(), v1.end(), x);
	if (p != v1.end())
	{
		 p = v1.insert(p, 100);
		 *(p + 1) *= 10;
		
	}

	print_container(v1);
}

运行结果:

在这里插入图片描述

  

8.3 总结

  用 i n s e r t insert insert 后,迭代器 p 就失效了最好的办法就是不再用 p p p 去访问。如果非要访问,一定要更新一下迭代器 p p p 再去访问

  
  

9 erase

iterator erase(iterator pos)
{
	assert(pos < end() && pos >= begin());

	iterator i = pos;
	while (i != end() - 1)
	{
		*(i) = *(i + 1);
		++i;
	}
	--_finish;

	return pos;
}

  同样, e r a s e erase erase 也存在迭代器失效的问题。我们使用的时候一定要注意! e r a s e erase erase之后就不要再访问迭代器,一定要访问也要先更新再访问,更新后的迭代器指向要删除数据的下一个位置
  
  

10 resize

void resize(size_t n, const T& val = T())
{
	if (n < size())
	{
		_finish = _start + n;
	}
	else
	{
		reserve(n);
		while (_finish < _start + n)
		{
			*_finish = val;
			++_finish;
		}
	}
}

   r e s i z e resize resize 的功能与实现这里就不再过多解释,这里大家直接看代码即可。

  这里我们重点讲一下 const T& val = T()的写法
  这是自定义类型给缺省值的方法。因为 v e c t o r vector vector 是模板,给缺省值时不能再像 const T& val = 0 这样给,因为是模板,你不知道 T 是什么类型,像前面给 0 就表示 T 一定是 0 了,肯定是不合适的。T 可能是 i n t int int,可能是 d o u b l e double double、也可能是自定义类型。

  如果是自定义类型的对象,我们就调用它的默认构造。上述给缺省值的方法就是给 T 的一个匿名对象并调用它的默认构造

  那如果 T 是内置类型 i n t int int d o u b l e double double怎么办?他们没有默认构造这一概念啊
  在此之前,我们确实是认为默认函数是没有构造函数这一概念的,但是为了兼容像模板这样的场景,内置类型也有了构造函数的概念(还有析构函数的概念)。

void test5()
{
	int i = int();
	int j(1);
	
	cout << i << endl;
	cout << j << endl;
	cout << int(2) << endl;
}

运行结果:

在这里插入图片描述

  
  

11 构造函数系列

11.1 默认构造

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

  但实际上,并不需要这么麻烦,因为我们在成员变量声明时,结了缺省值,我们这样就可以了

vector() {}

  学习类与对象时,我们知道:所有成员都会走初始化列表。如果我们没有显式写在初始化列表,对于内置类型:没有缺省值不做任何处理,但现在我们在成员变量声明时,结了缺省值,则用缺省值初始化
  
  在C++11中,给了一个方式可以 强制生成默认构造

vector() = default;

  因为编译器只有在程序员没有自己实现任何构造函数(拷贝构造也是构造)时才会生成默认构造,如果我们自己实现了带参的构造,又想编译器生成默认构造,就可以用上述语法
  
  

11.2 拷贝构造

vector(const vector<T>& v)
{
	reserve(v.capacity());

	for (auto& e : v)
	{
		push_back(e);
	}
}

  拷贝构造,就是遍历 对象 v v v,将 v v v 的值全部 p u s h push push_ b a c k back back t h i s this this 中。这里范围for用了引用,是为了减少拷贝,而为了避免频繁的扩容,使用 r e s e r v e reserve reserve 提前开辟空间

  
  

11.3 迭代器区间构造 与 n 个 value 构造

11.3.1 迭代器区间构造

  在stl库中,还提供了一个构造方法:迭代器区间构造

在这里插入图片描述

  这里讲一个新的知识点:类模板里面的成员函数可以是函数模板

为什么要用模板呢?这样不行吗?

vector(iterator first, iterator last)

  如果是这样的话,那么迭代器构造只能用 v e c t o r vector vector 迭代器构造,如果是函数模板的话可以用任意的迭代器进行构造
  
代码如下:

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

测试:

void test6()
{
	list<int> l1;
	l1.push_back(1);
	l1.push_back(2);
	l1.push_back(3);
	l1.push_back(4);
	l1.push_back(5);
	l1.push_back(6);

	vector<int> v1(++l1.begin(), --l1.end());
	print_vector(v1);
}

运行结果:

在这里插入图片描述

  可见,迭代器区间可以用任意容器迭代器进行初始化,但它要求类型是匹配的
  
  

11.3.2 n 个value构造

   n n n v a l u e value value 的构造很简单,不断 p u s h push push_ b a c k back back 就行

vector(size_t n, const T& val = T())
{
	reserve(n);

	while (n--)
	{
		push_back(val);
	}
}

  

11.3.3 注意事项

  调用 n n n v a l u e value value 的构造函数时,会出现一个很奇怪的报错

void test7()
{
	vector<int> v(10, 1);
	print_vector(v);
}

在这里插入图片描述
在这里插入图片描述

  
  为什么我调用的时 n n n v a l u e value value 的构造,结果报错报到迭代器区间的构造了呢?
  这是因为编译器把vector<int> v(10, 1);中的 10 和 1 识别成两个迭代器了。
  为什么会这样呢?编译器有一个原则,那就是匹配最匹配

  我们先来看 n n n v a l u e value value 的构造
  vector(size_t n, const T& val = T())对 10 和 1 来说,T 是 i n t int int 是确定的
  但是 n n n s i z e size size_ t t t i n t int int进行类型转换

  再看迭代器区间构造
  vector(InputIterator first, InputIterator last),它实例化出来是两个 i n t int int不用类型转换,所以编译器选择迭代器区间构造

  那怎么解决呢?
  stl库中选择重载多个 n n n v a l u e value value 的构造函数,让你永远有更好的选择

 vector(size_t n, const T& value = T())
 {
     reserve(n);
     while (n--)
     {
         push_back(value);
     }
 }
 
 vector(int n, const T& value = T())
 {
     reserve(n);
     while (n--)
     {
         push_back(value);
     }
 }
 
 vector(long long n, const T& value = T())
 {
     reserve(n);
     while (n--)
     {
         push_back(value);
     }
 }

  
  

12 赋值运算符重载

  赋值运算符重载分为现代写法传统写法,这在 s t r i n g string string 的模拟实现中讲过类似的,这里就不再细讲了,我们直接看代码
  

12.1 传统写法

void clear()
{
	_finish = _start;
}

vector<T>& operator=(const vector<T>& v)
{
	clear();

	if (this != &v)
	{
		reserve(v.capacity());
		const_iterator it = v.begin();
		while (it != v.end())
		{
			push_back(*it);
			++it;
		}
	}

	return *this;
}

  

12.2 现代写法

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> tmp)
{
	swap(tmp);
	return *this;
}

  
类里面可以用类名替代类型,类外面不行

  比如这样:用 类名vector 替代 类型vector<int>

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

  不过并不推荐这种写法,因为对新手来说不友好。
  
  

13 源码

//vector.h

#pragma once
#include<iostream>
#include<assert.h>
#include<vector>

namespace my_vector
{
    template<class T>
    class vector
    {
    public:
        // Vector的迭代器是一个原生指针
        typedef T* iterator;
        typedef const T* const_iterator;

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

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

        void clear()
        {
            _finish = _start;
        }
        
        
        vector()
            :_start(nullptr)
            , _finish(nullptr)
            , _end_of_storage(nullptr)
        {}
        
        // C++11 前置生成默认构造
		vector() = default;

        vector(size_t n, const T& value = T())
        {
            reserve(n);
            while (n--)
            {
                push_back(value);
            }
        }
        vector(int n, const T& value = T())
        {
            reserve(n);
            while (n--)
            {
                push_back(value);
            }
        }
        vector(long long n, const T& value = T())
        {
            reserve(n);
            while (n--)
            {
                push_back(value);
            }
        }

        vector(const vector<T>& v)
        {
            reserve(v.capacity());

            const_iterator it = v.begin();
            while (it != v.end())
            {              
                push_back(*it);
                it++;
            }
            
        }

        template<class InputIterator>
        vector(InputIterator first, InputIterator last)
        {
            InputIterator tmp = first;
            //不能是< , list那些就不行
            while (tmp != last)
            {
                push_back(*tmp);
                tmp++;
            }
        }

 		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> tmp)
        {
            swap(tmp);
            return *this;
        }

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

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

        void reserve(size_t n);
        void push_back(const T& x);
        void resize(size_t n, const T& value = T());
        void pop_back();
        iterator insert(iterator pos, const T& x);        
        iterator erase(iterator pos);

    private:
        iterator _start = nullptr; // 指向数据块的开始
        iterator _finish = nullptr; // 指向有效数据的尾
        iterator _end_of_storage = nullptr; // 指向存储容量的尾
    };

    template<class T>
    void vector<T>::push_back(const T& x)
    {
        if (size() == capacity())
        {
            size_t new_capacity = capacity() ? 2 * capacity() : 4;
            reserve(new_capacity);
        }
        
        *_finish = x;
        _finish++;
    }

    template<class T>
    void vector<T>::reserve(size_t n)
    {
        if (n > capacity())
        {   
            iterator tmp = new T[n];
            size_t preserve_size = size();

            int m = 0;
            iterator it = begin();
            while (it != end())
            {            
                *(tmp + m) = *it;
                ++m;
                ++it;
            }
            //memcpy(tmp, _start, sizeof(T) * size());

            delete[] _start;

            _start = tmp;
            _finish = _start + preserve_size;
            _end_of_storage = _start + n;
        }
    }

    template<class T>
    void vector<T>::resize(size_t n, const T& value)
    {
        if (n <= size())
        {
            _finish = _start + n;
        }

        else
        {
            reserve(n);

            int m = n - size();
            while (m--)
            {
                *_finish = value;
                ++_finish;
            }
        }
    }

    template<class T>
    void vector<T>::pop_back()
    {
        assert(!empty());
        _finish;
    }

    /
    template<class T>
    typename vector<T>::iterator vector<T>::insert(typename vector<T>::iterator pos, const T& x)
    {
        assert(pos <= _finish && pos >= _start);
        /*if (size() == capacity())
        {
            size_t new_capacity = capacity() ? 2 * capacity() : 4;
            reserve(new_capacity);
        }*/
        //相对位置
        if (size() == capacity())
        {
            size_t len = pos - _start;
            reserve(capacity() ? 2 * capacity() : 4);
            pos = _start + len;
        }

        iterator it = end();
        while (it > pos)
        {
            *it = *(it - 1);
            it--;
        }
        *pos = x;
        ++_finish;

        return pos;
    }

    template<class T>
    typename vector<T>::iterator vector<T>::erase(typename vector<T>::iterator pos)
    {
        assert(pos < end() && pos >= begin());

        iterator it = pos + 1;
        while (it != end())
        {
            *(it - 1) = *it;
            ++it;
        }
        --_finish;

        return pos;
    }

    template<class T>
    void print_vector(const vector<T>& v)
    {
        typename vector<T>::const_iterator it = v.begin();
        while (it != v.end())
        {
            std::cout << *it << " ";
            ++it;
        }
        std::cout << std::endl;
    }

    template<class Container>
    void print_container(const Container& v)
    {
        auto it = v.begin();
        while (it != v.end())
        {
            std::cout << *it << " ";
            ++it;
        }
        std::cout << std::endl;
    }
}


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2121266.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

打破界限,自闭症寄宿学校带给孩子的改变

在社会的广阔画卷中&#xff0c;有一群特别的孩子&#xff0c;他们以独特的视角感知世界&#xff0c;以非凡的方式表达情感&#xff0c;他们就是自闭症儿童。自闭症&#xff0c;这个听起来略带神秘色彩的词汇&#xff0c;实则承载着无数家庭的期盼与挑战。在这片充满爱的土地上…

【北京迅为】《STM32MP157开发板使用手册》-第十六章 Buildroot制作根文件系统

iTOP-STM32MP157开发板采用ST推出的双核cortex-A7单核cortex-M4异构处理器&#xff0c;既可用Linux、又可以用于STM32单片机开发。开发板采用核心板底板结构&#xff0c;主频650M、1G内存、8G存储&#xff0c;核心板采用工业级板对板连接器&#xff0c;高可靠&#xff0c;牢固耐…

丰巢“闯关”港交所上市

社区中随处可见的智能快递柜&#xff0c;即将捧出一个IPO。 近日&#xff0c;丰巢控股有限公司&#xff08;下称“丰巢控股”或“丰巢”&#xff09;正式向港交所递交了招股书&#xff0c;华泰国际担任其独家保荐人。这将是继顺丰控股、顺丰房托、嘉里物流、顺丰同城之后&…

【Qt笔记】QGroupBox控件详解

目录 引言 一、基本属性 二、常用方法 2.1 构造函数 2.2 设置标题 2.3 设置复选框模式 2.4 是否被选中 2.5 设置对齐方式 2.6 设置扁平化样式 三、信号与槽机制 四、样式定制 五、应用示例 5.1 代码 5.2 代码解析 5.3 实现效果 结语 引言 QGroupBox 是 Qt…

检查iOS多语系文件内容检查iOS多语系文件内容

在iOS中&#xff0c;检查多语言文件&#xff08;如 .strings 文件&#xff09;内容的命令通常使用 plutil 工具。你可以通过终端执行以下命令来检查 .strings 文件的格式和内容&#xff1a; plutil -lint path/to/your/Localizable.strings 这个命令会验证指定的 .strings 文…

C语言13--结构体

结构体基本概念 C语言提供了众多的基本类型&#xff0c;但现实生活中的对象一般都不是单纯的整型、浮点型或字符串&#xff0c;而是这些基本类型的综合体。比如一个学生&#xff0c;典型地应该拥有学号&#xff08;整型&#xff09;、姓名&#xff08;字符串&#xff09;、分数…

已配置好的Linux CentOS7虚拟机转换为可视化界面问题

一、发现问题 学习过程中发现可视化界面比较有意思&#xff0c;就想尝试搞一下看看&#xff0c;于是去网站上搜索&#xff0c;看到的一些是在新建虚拟机的时候进行设置的&#xff0c;我尝试跟着步骤去搞&#xff0c;发现其中最关键的一步&#xff0c;软件选择中&#xff0c;没有…

【北京迅为】《STM32MP157开发板使用手册》- 第十五章 制作最小linux系统

iTOP-STM32MP157开发板采用ST推出的双核cortex-A7单核cortex-M4异构处理器&#xff0c;既可用Linux、又可以用于STM32单片机开发。开发板采用核心板底板结构&#xff0c;主频650M、1G内存、8G存储&#xff0c;核心板采用工业级板对板连接器&#xff0c;高可靠&#xff0c;牢固耐…

TikTok内容电商:短视频与直播带货如何重塑消费者购物决策

数字化时代&#xff0c;内容电商已经成为一种重要的商业模式。而TikTok作为全球领先的短视频平台&#xff0c;其内容电商模式正慢慢改变用户的消费习惯。TikTok Shop作为TikTok平台上的电商板块&#xff0c;也凭借其独特的短视频和直播带货模式&#xff0c;影响着消费者的购物决…

百度MEG数据开发治理平台-TDS

导读 百度MEG的上一代大数据产品存在平台分散、质量不均和易用性差等问题&#xff0c;导致开发效率低下、学习成本高&#xff0c;业务需求响应迟缓。为了解决这些问题&#xff0c;百度MEG内部开发了图灵3.0生态系统。图灵3.0覆盖了数据全生命周期&#xff0c;包括Turing Data …

AI在医学领域:HMARL首个多器官诊断AI框架

多器官疾病因其对多个器官系统的同时影响而带来了显著的挑战&#xff0c;这需要复杂和适应性的治疗策略。尽管在人工智能驱动的医疗决策支持系统方面取得了最新进展&#xff0c;但现有的解决方案通常限于单个器官系统。它们往往忽视了器官系统之间复杂的相互依赖性&#xff0c;…

搜维尔科技:SenseGlove触觉反馈数据手套为人形机器人遥操作提供精确的控制和交互方案

SenseGlove触觉反馈数据手套 使用市场上唯一一款结合力反馈、振动触觉反馈和运动捕捉以及紧凑无线设计的触觉手套来收集数据。 遥操作机器人 远程机器人向人类提供触觉反馈&#xff0c;提供更强的真实感和更高的性能&#xff0c;以及安全性和控制力。远程机器人的 SenseGlov…

适用于BLE室内定位系统的自适应路径损耗模型

自适应路径损耗模型(ADAM):提升BLE室内定位精度的创新方法 室内定位系统(IPS)在物联网、智慧城市等领域中扮演着至关重要的角色。然而,由于室内环境的复杂性(如信号多径效应、障碍物等),传统的定位方法往往面临精度不足的问题。本文介绍了一种新颖的模型——ADAM(Ad…

【毕设项目五】基于SpringBoot+VUE的公共卫生教育与宣传系统

基于SpringBootVUE的公共卫生教育与宣传系统 项目介绍 系统有两种角色&#xff1a;管理员和普通用户。 &#xff08;1&#xff09;健康教育资源 &#xff08;2&#xff09;活动管理 &#xff08;3&#xff09;反馈与建议 &#xff08;4&#xff09;用户管理 管理员实现功…

快速上手指南:在Windows系统中下载Ollama,一键启动大模型体验!

1. 下载ollama 官网下载安装&#xff1a; ollama.com 2. 拉取大模型 llama 3.1 终端中输入 ollama pull llama3.1&#xff0c;等待安装 3. 运行 llama3.1 ollama run llama3.1接下来就可以和模型对话了 退出 /bye运行 /? 查看更多聊天中命令 其他 ollama github&#x…

150Kg载重履带式无人车底盘技术详解

150Kg载重履带式无人车底盘&#xff0c;作为现代智能移动平台的重要组成部分&#xff0c;专为复杂地形作业设计&#xff0c;如野外勘探、灾难救援、农业自动化及军事侦察等领域。该底盘集成了先进的动力技术、稳定的履带行走系统、精准的遥控与控制系统以及模块化的设计理念&am…

[数据集][目标检测]河道垃圾检测数据集VOC+YOLO格式2274张8类别

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;2274 标注数量(xml文件个数)&#xff1a;2274 标注数量(txt文件个数)&#xff1a;2274 标注…

【安全知识】访问控制模型DAC、MAC、RBAC、ABAC有什么区别?

不同的公司或软件提供商&#xff0c;设计了无数种控制用户访问功能或资源的方法。但无论哪种设计&#xff0c;都可归到四种经典权限模型里——自主访问控制(DAC, Discretionary Access Control)、强制访问控制(MAC, Mandatory Access Control)、基于角色访问控制(RBAC, Role-ba…

SprinBoot+Vue高校就业管理系统设计与实现的设计与实现

目录 1 项目介绍2 项目截图3 核心代码3.1 Controller3.2 Service3.3 Dao3.4 application.yml3.5 SpringbootApplication3.5 Vue 4 数据库表设计5 文档参考6 计算机毕设选题推荐7 源码获取 1 项目介绍 博主个人介绍&#xff1a;CSDN认证博客专家&#xff0c;CSDN平台Java领域优质…

C语言代码练习(第十九天)

今日练习&#xff1a; 52、有一个已经排好序的数组&#xff0c;要求输入一个数后&#xff0c;按原来排序的规律将它插入数组中 53、输出"魔方阵"。所谓魔方阵是指它的每一行&#xff0c;每一列和对角线之和均相等。 54、找出一个二维数组中的鞍点&#xff0c;即该位置…