当前位置: 首页 > news >正文

c++-vector

文章目录

  • 前言
  • 一、vector介绍
  • 二、vector使用
    • 1、构造函数
    • 2、vector 元素访问
    • 3、vector iterator 的使用
    • 4、vector 空间增长问题
    • 5、vector 增删查改
    • 6、理解vector<vector< int >>
    • 7、电话号码的字母组合练习题
  • 三、模拟实现vector
    • 1、查看STL库源码中怎样实现的vector
    • 2、实现vector
    • 3、vector深浅拷贝问题


前言


一、vector介绍

  1. vector是表示可变大小数组的序列容器。
  2. 就像数组一样,vector也采用的连续存储空间来存储元素。也就是意味着可以采用下标对vector的元素进行访问,和数组一样高效。但是又不像数组,它的大小是可以动态改变的,而且它的大小会被容器自动处理。
  3. 本质讲,vector使用动态分配数组来存储它的元素。当新元素插入时候,这个数组需要被重新分配大小为了增加存储空间。其做法是,分配一个新的数组,然后将全部元素移到这个数组。就时间而言,这是一个相对代价高的任务,因为每当一个新的元素加入到容器的时候,vector并不会每次都重新分配大小。
  4. vector分配空间策略:vector会分配一些额外的空间以适应可能的增长,因为存储空间比实际需要的存储空间更大。不同的库采用不同的策略权衡空间的使用和重新分配。但是无论如何,重新分配都应该是对数增长的间隔大小,以至于在末尾插入一个元素的时候是在常数时间的复杂度完成的。
  5. 因此,vector占用了更多的存储空间,为了获得管理存储空间的能力,并且以一种有效的方式动态增长。
  6. 与其它动态序列容器相比(deque, list and forward_list), vector在访问元素的时候更加高效,在末尾添加和删除元素相对高效。对于其它不在末尾的删除和插入操作,效率更低。比起list和forward_list统一的迭代器和引用更好。

二、vector使用

1、构造函数

可以看到vector有下面的几种构造函数。并且还有一个构造函数模板。
在这里插入图片描述

void test01()
{//创建一个存储int类型的vector容器//调用的是explicit vector(const allocator_type& alloc = allocator_type())函数//此时v1里面没有内容vector<int> v1;cout << v1.capacity() << endl;//调用的是explicit vector(size_type n, const value_type& val = value_type(), const allocator_type& alloc = allocator_type())函数//将vector中初始化为10个1。vector<int> v2(10, 1);cout << v2.size() << endl;cout << v2.capacity() << endl;//还可以使用迭代器来进行构造,此时就会使用//template<class InputIterator>//vector(InputIterator first,InputIterator last, const allocator_type& alloc = allocator_type())模板生成对应的迭代器构造函数。//此时v3的内容就是v2里面的内容vector<int> v3(v2.begin(), v2.end());cout << v3.size() << endl;cout << v3.capacity() << endl;for (auto n : v3){cout << n << " ";}cout << endl;//还可以使用其它类类型对象的迭代器来进行构造。string s("hello world");//此时v4中都为int元素,所以保存的是每个字符对应的ascii码值。vector<int> v4(s.begin(), s.end());cout << v4.size() << endl;cout << v4.capacity() << endl;for (auto n : v4){cout << n << " ";}cout << endl;//使用拷贝构造函数来进行初始化vector<int> v5(v4);cout << v5.size() << endl;cout << v5.capacity() << endl;for (auto n : v5){cout << n << " ";}cout << endl;}

2、vector 元素访问

在这里插入图片描述

void test06()
{vector<int> v1;v1.push_back(1);v1.push_back(2);v1.push_back(3);v1.push_back(4);v1.push_back(5);//使用[]访问v1的元素cout << v1[0] << endl;//使用at访问v1的元素cout << v1.at(0) << endl;//front返回v1的头元素cout << v1.front() << endl;//back返回v1的尾元素cout << v1.back() << endl;//返回存储v1的数据的地址cout << v1.data()[0] << endl;}

3、vector iterator 的使用

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

void test02()
{vector<int> v1;v1.push_back(1);v1.push_back(2);v1.push_back(3);v1.push_back(4);//想要变量vector容器内的元素,可以使用下标来遍历for (size_t i = 0; i < v1.size(); ++i){cout << v1[i] << " ";}cout << endl;//也可以使用范围for来遍历for (auto e : v1){cout << e << " ";}cout << endl;//当然也可以使用迭代器来进行遍历//下面为使用正向迭代器正序遍历v1里面的元素,vector<int>::iterator it = v1.begin();while (it != v1.end()){//因为该迭代器没有被const修饰,所以也可以更改v1的内容*it = 2 * (*it);cout << *it << " ";++it;}cout << endl;//下面为使用反向迭代器倒序遍历v1里面的元素vector<int>::reverse_iterator rit = v1.rbegin();while (rit != v1.rend()){//因为该迭代器没有被const修饰,所以也可以更改v1的内容*rit = 2 * (*rit);cout << *rit << " ";++rit;}cout << endl;//下面为使用const修饰的正向迭代器正序遍历v1里面的元素vector<int>::const_iterator cit = v1.cbegin();while (cit != v1.cend()){//因为该迭代器被const修饰了,所以只能读取v1数据,不能修改v1的数据//*cit = 2 * (*cit);   //错误,不能修改v1的元素的值cout << *cit << " ";++cit;}cout << endl;//下面为使用const修饰的反向迭代器倒序遍历v1里面的元素vector<int>::const_reverse_iterator crit = v1.crbegin();while (crit != v1.crend()){//因为该迭代器被const修饰了,所以只能读取v1数据,不能修改v1的数据//*crit = 2 * (*crit);   //错误,不能修改v1的元素的值cout << *crit << " ";++crit;}cout << endl;
}

4、vector 空间增长问题

在这里插入图片描述

void test03()
{vector<int> v1(20, 1);vector<int> v2;//size()获取v1的数据个数cout << v1.size() << endl;//max_size()获取vector存储的最大元素个数//因为int型元素占4个字节,所以只能存10亿多个元素,而char类型元素占4个字节,所以存40亿多个元素cout << v1.max_size() << endl;//capacity()获取v1的容量大小cout << v1.capacity() << endl;//empty()判断v1、v2是否为空cout << v1.empty() << endl;cout << v2.empty() << endl;//shrink_to_fit()缩容函数,将v3的capacity缩容。vector<int> v3(100);cout << v3.capacity() << endl;v3.resize(10);cout << v3.capacity() << endl;//会将v3的capacity缩容到合size一样的大小v3.shrink_to_fit();cout << v3.capacity() << endl;
}

vector容器中的resize()函数和reserve()函数的区别和string类中的两个函数的区别类似。
reserve只负责开辟空间,如果确定知道需要用多少空间,reserve可以缓解vector频繁增容的代价缺陷问题。
resize在开空间的同时还会进行初始化,影响size。
在这里插入图片描述
在这里插入图片描述

void test04()
{//使用reserve开辟空间,只会改变capacity的值,不会进行初始化,所以不会改变size的值。vector<int> v1;v1.reserve(10);cout << v1.size() << endl;cout << v1.capacity() << endl;v1.push_back(1);v1.push_back(2);v1.push_back(3);v1.push_back(4);//当使用reserve开辟的空间没有原来的容量大时,就不会做任何处理v1.reserve(2);cout << v1.size() << endl;cout << v1.capacity() << endl;for (auto e : v1){cout << e << " ";}cout << endl;//使用resize开辟空间,第二个参数为新空间要初始化的值vector<int> v2;v2.resize(10,1);cout << v2.size() << endl;cout << v2.capacity() << endl;for (auto e : v2){cout << e << " ";}cout << endl;//使用resize开辟空间,如果不传第二个参数,默认将新空间的值初始化为0。vector<int> v3;v3.resize(10);cout << v3.size() << endl;cout << v3.capacity() << endl;for (auto e : v3){cout << e << " ";}cout << endl;//使用resize减少数据//当使用resize(n)开辟的空间没有当前的容量大时,resize会将容器内的元素删除到为n,此时size也会被改为nvector<int> v4(10, 1);v4.resize(5);cout << v4.size() << endl;cout << v4.capacity() << endl;for (auto e : v4){cout << e << " ";}cout << endl;
}

5、vector 增删查改

在这里插入图片描述

void test05()
{//assign为重新向容器中分配值vector<int> v1(5, 1);cout << v1.size() << endl;cout << v1.capacity() << endl;for (auto e : v1){cout << e << " ";}cout << endl;//使用assign(5,2)后会将v1原来的数据给清除,然后重新分配新的数据进去。v1.assign(5, 2);cout << v1.size() << endl;cout << v1.capacity() << endl;for (auto e : v1){cout << e << " ";}cout << endl;//只有当assign传入的大小大于原来的容量时,才会进行扩容。v1.assign(10, 3);cout << v1.size() << endl;cout << v1.capacity() << endl;for (auto e : v1){cout << e << " ";}cout << endl;//push_back为尾插一个元素v1.push_back(4);for (auto e : v1){cout << e << " ";}cout << endl;//pop_back为尾删一个元素v1.pop_back();for (auto e : v1){cout << e << " ";}cout << endl;//insert为在任意位置插入一个元素,使用迭代器确定位置//在v1的第三个位置之后插入5。v1.insert(v1.begin()+3, 5);//在v1的第五个位置之后插入3个6v1.insert(v1.begin() + 5, 3, 6);for (auto e : v1){cout << e << " ";}cout << endl;//erase为删除任意位置的数据,使用迭代器确定位置//如果不传入结束的位置,则就会将v1的全部数据删除。//v1.erase(v1.begin());//将v1的第3个位置之后,包括第5个位置的数据删除。即(3,5]的数据删除。v1.erase(v1.begin() + 3, v1.begin() + 5);for (auto e : v1){cout << e << " ";}cout << endl;//swap为交换两个vector容器的数据vector<int> v2(10, 1);vector<int> v3(10, 2);for (auto e : v2){cout << e << " ";}cout << endl;for (auto e : v3){cout << e << " ";}cout << endl;v2.swap(v3);for (auto e : v2){cout << e << " ";}cout << endl;for (auto e : v3){cout << e << " ";}cout << endl;//clear为将vector容器中的数据都清除vector<int> v4(10, 1);for (auto e : v4){cout << e << " ";}cout << endl;v4.clear();for (auto e : v4){cout << e << " ";}cout << endl;//
}

vector中没有find函数,但是可以通过算法模块实现。
在这里插入图片描述

//find函数为查找,这个是算法模块实现,不是vector的成员接口vector<int> v5;v5.push_back(1);v5.push_back(2);v5.push_back(3);v5.push_back(4);//使用这个find,当找到元素时会返回指向该元素的迭代器vector<int>::iterator ret01 = find(v5.begin(),v5.end(),3);cout << *ret01 << endl;

6、理解vector<vector< int >>

当我们写下面的题时,发现使用c++写时,题目中给了我们一个vector<vector< int >>的返回值。
在这里插入图片描述
vector<vector< int >> vv 表示的就是vv中的每个元素都是vector< int >类型的。
在这里插入图片描述

7、电话号码的字母组合练习题

题目链接
在这里插入图片描述

三、模拟实现vector

1、查看STL库源码中怎样实现的vector

我们可以在DevC++的文件目录下找到stl库的源码文件。具体路径如下:

D:\Dev-Cpp\MinGW64\lib\gcc\x86_64-w64-mingw32\4.9.2\include\c++\bits

在bits这个文件夹下面我们看到stl库的源码。
在这里插入图片描述
在linux系统下,我们可以在/usr/include/c++/4.8.2/bits目录下看到stl库的源码。gcc使用的是SGI版本的STL库。

cd /usr/include/c++/4.8.2/bits

在这里插入图片描述
我们看到stl的源码中有三个成员变量分别为:
在这里插入图片描述
stl的源码中就是使用这三个成员变量来求size和capacity等一些值。

2、实现vector

因为查看源码时看到源码的vector使用了start和finish和end _ of _ storage这三个指针。所以我们模拟实现vector也靠这些指针。
template< class T >为一个模板,因为vector容器里面可以存任意类型的数据,可以是内置类型,也可以是自定义类型,所以我们使用模板来实现。T就相当以后vector里面的数据类型,可能是int、char、string类、Date类或者vector< int >等类型。
下面就是我们模拟实现的vector的刚开始的模板。

在这里插入图片描述
我们先实现push_back尾插函数。

在这里插入图片描述
但是因为push_back插入元素要考虑扩容的问题,所以我们要先实现reserve函数。又因为reserve函数需要用到size和capacity,所以我们要先实现size和capacity函数。下面为size和capacity函数的实现。
在这里插入图片描述
在这里插入图片描述
然后我们再来实现reserve函数。
在这里插入图片描述
接下来我们实现vector的[]操作符的重载函数。

在这里插入图片描述
然后我们进行测试时会发现出现了异常,
在这里插入图片描述
我们调试后发现在reserve函数中,当申请了一片新空间后,_start和_end_of_storage的值都改变了,而_finish的值还是nullptr,所以在push_back函数中解引用_finish时才出现了空指针解引用的异常。这个异常是因为我们在reserve中求_finish时调用了size函数,而此时_start已经变为了tmp,此时_finish = tmp + _finish - tmp,所以_finish还是为nullptr,这才出现了异常。想要解决这个异常有两种方法。

在这里插入图片描述
在这里插入图片描述
(1). 交换语句顺序。(不推荐,以后不好维护代码,顺序反了程序就崩溃)

在这里插入图片描述
(2). 提前将size的值算出来。(推荐)
在这里插入图片描述

接下来我们就实现vector的迭代器中的begin和end。
在这里插入图片描述
当实现了begin和end后,就可以使用迭代器和范围for来遍历vector的元素了。
在这里插入图片描述
但是此时我们发现如果是const修饰的vector的对象,此时没有办法调用[]操作符重载函数,也没有办法调用迭代器的begin和end函数等。
在这里插入图片描述
所以我们还需要写一个const修饰的[]操作符重载函数和迭代器。
在这里插入图片描述
在这里插入图片描述

然后我们再来实现pop_back方法了。在实现pop_back函数时,因为可能会遇到vector为空的情况,所以我们需要写一个empty函数来判断vector是否为空。并且在pop_back函数中使用assert断言vector是否为空。

在这里插入图片描述
如果不判断vector是否为空,就会出现如下的错误。使用迭代器遍历vector的元素时会一直循环下去,这时因为此时_start在_finish的后面了,所以会一直向后访问下去。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

下面再来实现resize函数。我们看到官方库中的resize函数的第二个参数为缺省参数value_type val = value_type()。因为vector中可以存放任意类型的数据,例如内置类型中的int、char、double等,自定义类型的Date、string类等。所以就不能给缺省参数赋值时给一个定值,而value_type val = value_type()为创建一个匿名对象,在创建匿名对象时会调用他的默认构造函数,然后给val的缺省值就为这个匿名对象的值,这样来实现给val初始化。
在这里插入图片描述
那么我们的内置类型int、double、char有构造函数吗?
因为有了模板的存在,所以这些内置类型也有构造函数。但是指针类型不支持这样显示的调用构造函数。
在这里插入图片描述
在这里插入图片描述
但是指针类型还是支持使用模板调用它的构造函数。例如下面T为int * 时,int * = int * (),使用模板就可以进行int类型的构造函数。
在这里插入图片描述
所以我们自己实现resize函数时,将resize的第二个参数也写为T val = T()的形式。然后我们进行测试也没有出现问题。
在这里插入图片描述
在这里插入图片描述
接下来我们实现insert函数。
在这里插入图片描述
当我们测试时我们发现当vector中有4个元素时,此时再向vector中插入元素会出现错误,而当vector中有5个元素时,此时再向vector中插入元素不会出错。这其实是因为发生了迭代器失效问题。
在这里插入图片描述
我们可以看到当vector中有4个元素后,此时再向vector中插入元素会调用reserve函数来进行扩容。而当扩容后我们看到_start、_finish、_end_of_storage的值都发生了变化,即指向了扩容后的空间的地址,而pos的值没有变化,即pos还指向了原来的空间的地址。此时pos指向的空间已经被释放,所以pos此时就相当于一个野指针,指向已经被释放的空间。而我们使用
pos=val修改的是原来空间的值,而新的空间并没有插入val,但是_finish已经+1向后移动一位了,所以vector的最后一个元素为随机值。上述的这种情况为最常见的迭代器失效问题,类似于野指针问题。
在这里插入图片描述

在这里插入图片描述
没扩容之前_start、_finish、_end_of_storage、pos都指向同一片空间。
在这里插入图片描述
扩容之后_start、_finish、_end_of_storage指向新的空间,而_pos还指向原来的空间。
在这里插入图片描述
当执行*pos=val后,原来空间中的值变了,而指向新空间的_finish+1向后移动了一位。
在这里插入图片描述
解决办法:更新pos。
即先求出pos和_start之间的距离,当扩容后,更新pos的指向,即让pos也指向新的空间。
在这里插入图片描述

当我们将代码修改了后,可以解决上面的问题,但是又会出现下面的两种情况的迭代器失效问题。
(1) insert里面没有发生扩容,但是pos指向了新插入的元素,而不是原来的3了,是迭代器失效。
我们在测试代码中使用find函数找到vector中3元素的位置,因为find函数为std模板生成的函数,所以返回的是一个迭代器,即返回的是指向3的一个迭代器。但是当我们向pos位置插入30元素后,此时再(*pos)++,会发现是新插入的元素30++变为31了,而不是3++变为4。这也是迭代器失效问题,因为pos指向的是元素3,但是(*pos)++后3没有变为4,而是30变为31了。
在这里插入图片描述
(2) insert里面发生了扩容,insert里面的pos发生了改变,但是测试代码里面的pos没有发生改变。也是迭代器失效。
我们发现当在insert里面发生了扩容之后,在insert里面已经更新了pos,但是测试代码里面的pos还没有变。因为我们调用insert时是值传递,所以insert中修改的为pos的拷贝,而测试代码中的pos并没有改变。所以(*pos)++是将pos指向的原来的空间里的数据++了。

在这里插入图片描述
那么我们可以将insert修改为传引用传参。

在这里插入图片描述
在这里插入图片描述
传引用传参虽然解决了上面的问题,但是当我们直接向insert中传入begin和end时又会出错,这是因为begin中是传值返回,传值返回会发生拷贝,而拷贝生成的临时变量具有常性。所以不能使用传引用来修改。

在这里插入图片描述

在这里插入图片描述
我们查看st文档可以看到insert函数有一个iterator的返回值,即库里面的解决办法是传回新的pos迭代器。当使用完insert后,将测试代码的pos赋值为insert的返回值即可更新pos。所以我们也使用这样的方法。std库里面的迭代器在insert后,如果在insert里面进行了扩容,而没有将pos接收insert的返回值从而更新pos迭代器时,下面再使用pos也会出现错误。这时就需要使用pos = v1.insert(pos, 30);更新pos迭代器,然后才不会出错。但是insert以后,我们就认为pos失效了,不能再使用。
在这里插入图片描述
在这里插入图片描述

下面我们再来进行erase函数的实现。
在这里插入图片描述
在这里插入图片描述
我们实现的erase,在erase之后,pos迭代器没有失效,但是std库里面的pos在erase之后会失效。即在windows下的VS中会中止程序。
在这里插入图片描述
在这里插入图片描述
但是同样的代码在Linux下的g++中不会中止程序。erase g++的实现和我们的实现类似。
在这里插入图片描述
在这里插入图片描述

那么erase之后,我们认为pos失效吗?
下面为删除的是最后一个元素。VS会报错。
在这里插入图片描述
在这里插入图片描述
但是在Linux下的g++中还是可以运行,但是此时pos迭代器指向的位置已经越界了,是不应该被访问的。
在这里插入图片描述
在这里插入图片描述

结论:erase之后,pos失效了,不要访问,行为结果未定义。因为不同编译器下的结果不同。
所以insert之后pos不要访问,因为pos可能为野指针。erase之后pos也不要访问,因为可能pos指向的位置可能越界。

下面我们使用erase来进行一个练习,删除所有的偶数。
在window下的VS中,我们使用下面的代码来进行删除所有的偶数,在使用erase之后,VS编译器会强制检查,如果it没有更新则就会出错。
在这里插入图片描述
在这里插入图片描述

同样的代码在Linux下的g++中运行的情况如下。
当最后一个数为奇数,但是没有进行it迭代器更新时,在g++中可以正常执行,因为g++中不会进行强制检查。
在linux中使用g++编译时,如果代码中使用了c++11的语法,就要在编译时加上-std=c++11,即按照c++11的语法编译。

-std=c++11

在这里插入图片描述
在这里插入图片描述
但是当最后一个数为偶数时,程序会出现段错误。
在这里插入图片描述
在这里插入图片描述
我们经过下面的分析后发现是因为我们的删除逻辑不对。
在这里插入图片描述
那么我们将判断条件改为it<v1.end()。it就为图中的pos,v1.end()就为图中的finish。此时发现程序没有出现错误。但是这样的改法并不能真正的解决问题。
在这里插入图片描述
在这里插入图片描述
如果我们有连续的偶数在一起时,此时发现有的偶数没有被删除。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

我们知道上面的问题都是因为rease之后迭代器失效问题和我们的删除逻辑有问题,所以我们在使用erase之后需要将迭代器更新,而c++的文档中也写了erase函数返回的就是最新的迭代器,所以我们在每次使用完erase之后需要进行迭代器更新。
在这里插入图片描述
当我们将代码改为这样时,就可以实现删除偶数了。
下面为在VS下可以正常删除偶数了。
在这里插入图片描述
此时在linux下的g++中程序也正常运行。
在这里插入图片描述
在这里插入图片描述
所以我们将自己写的erase也改为将新的pos返回。
在这里插入图片描述
然后我们实现vector的析构函数。
在这里插入图片描述

然后我们实现构造函数中的将n个元素初始化为val的构造函数。我们知道第二个缺省参数val为一个匿名对象的引用,但是我们之前说过匿名对象生命周期只在这一行,那么在函数中val引用的匿名对象不就销毁了吗?

在这里插入图片描述
我们看到匿名对象A()的生命周期只在那一行。
这是因为当这行之后没有人会使用这个匿名对象了,所以这个匿名对象的生命周期只在当前一行。
在这里插入图片描述
当使用const修饰的引用指向这个匿名对象时,这个匿名对象就不会在这一行之后销毁,会随着xx的销毁而销毁。
const引用会延长匿名对象的生命周期到引用对象域结束,因为以后使用xx就代表匿名对象。
在这里插入图片描述
所以我们将下面的构造函数这样写。并且在写这个构造函数时记得初始化,因为此时this里面的值都为随机值,如果不初始化,那么在reserve中求的capacity和size都是不对的,所以要记得将_start、_finish、_end_of_storage初始化。
在这里插入图片描述

我们看到在c++文档中还有一个这样的构造函数模板。这个相当于一个迭代器区间初始化的模板。这里面为什么不使用iterator而使用InputIterator,是因为这个迭代器区间不是必须要用vector的迭代器区间,它可以使用一个string类类型对象的迭代器区间来进行初始化。可以看到可用s1的迭代器区间来初始化v1,所以不能写iterator,因为iterator只是vector里面的类型的迭代器。
在这里插入图片描述
在这里插入图片描述
所以我们自己实现迭代器区间初始化的构造函数时也写一个模板。然后我们将迭代器区间的内容都push_back到vector中。
在这里插入图片描述
当我们写完这个模板后,测试时发现出现了下面的非法的间接寻址错误。这是因为当没有template < class InputIterator >模板时,vector< int > v1(10,5)会去匹配vector(size_t n, const T& val = T())这个构造函数,然后此时int类型的10会进行类型转换变为size_t类型。而当有了template < class InputIterator >模板后,vector< int > v1(10,5)会去匹配template < class InputIterator >模板生成的构造函数,因为这个函数不会进行类型转换,编译器认为是最匹配的。所以就会出现将int类型的变量当作地址来进行寻址,然后就出现了非法的间接寻址错误。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
我们可以使用下面的两种方法来解决这个错误。
解决办法1:在调用vector(size_t n, const T& val = T())构造函数时,使用vector< int > v(10u, 5);这样的形式调用,因为加上u后10就表示无符号整数,就不会发生类型转换,而当有一个匹配的函数时,编译器就不会再根据template < class InputIterator >模板再生成函数了。

在这里插入图片描述
解决办法2:提供一份更合适的vector(size_t n, const T& val = T())构造函数的重载版本。这样vector< int > v(10, 5);就会调用vector(int n, const T& val = T())构造函数了。
在这里插入图片描述
在这里插入图片描述
当我们实现了template < class InputIterator >这个模板之后,我们也可以使用其它类型的迭代器区间来初始化vector的内容。
在这里插入图片描述

其实不只是在vector中使用了template < class InputIterator >这样的迭代器区间初始化模板,在sort中也使用了类似的方法来实现可以接收任意类型的迭代器来调用sort方法。
在这里插入图片描述
在这里插入图片描述
sort函数默认是升序,如果不传第三个参数默认就是升序。当我们创建一个greater< int >类型的对象当作第三个参数传入sort时,此时sort为降序排序。
在这里插入图片描述
在这里插入图片描述

3、vector深浅拷贝问题

当实现了vector上面的一些功能后,我们再来看看拷贝构造函数,我们看到编译器自动生成的拷贝构造函数为浅拷贝,即将v1和v2指向了同一片空间,这样会出现两次析构函数的调用,所以会出现错误。
在这里插入图片描述
我们可以自己写拷贝构造函数。
在这里插入图片描述
在这里插入图片描述
我们上面写的拷贝构造函数可以将vector< int >类型的对象进行正确的拷贝,但是当遇到vector< std::string >类型的对象时,就会出现错误。这是因为vector里面的string类类型的对象拷贝时也涉及到深浅拷贝问题,而使用memcpy(_start,v._start,sizeof(T)*v.size())对于string类类型对象来说是浅拷贝,所以此时v3和v4的里面存的是同一个string类类型对象。可以看到下面v3和v4中的第一个元素都是string类类型的对象,这两个string类类型的对象中_Ptr相等,即这两个string类类型对象指向了同一个字符串。所以v3和v4中的每个string类类型对象会调用两次析构函数,所以会出现错误。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
出现这个错误的原因是我们在拷贝构造函数中使用的memcpy函数来进行的拷贝,所以我们需要将memcpy浅拷贝换成深拷贝函数,所以我们使用自定义类型的赋值运算符重载函数,因为string类的赋值运算符重载函数是深拷贝,它会新开辟一片空间然后复制原来的字符串。
在这里插入图片描述
在这里插入图片描述

当我们向v4中插入元素,让v4进行扩容时,此时又会出现错误,因为reserve扩容函数中使用的数据拷贝也是memcpy,所以我们也需要将reserve中的memcpy浅拷贝函数换为自定义类型的赋值运算符重载函数,因为赋值运算符重载函数是深拷贝。此时新的空间中的string类类型对象的_str还指向已经释放的空间里面的内容,所以会出现错误。

在这里插入图片描述
在这里插入图片描述
我们将reserve函数里面的memcpy函数也改变后,就不会出现错误了。
在这里插入图片描述
在这里插入图片描述

但是此时我们测试杨辉三角类时,又出现了错误。即我们创建一个vector< vector< int > >用来接收Solution类的成员函数generate返回的一个vector< vector< int > >数组。这是因为我们在拷贝时使用了赋值运算符重载函数,但是我们并没有重写vector的赋值运算符重载函数,所以vector使用默认生成的赋值运算符重载函数,默认生成的为浅拷贝,所以在执行< vector < vector< int > > ret = Solution().generate(5)时,虽然ret和vv的_start、_finish、_end_of_storage指向了不同的空间,但是因为将ret[0] = vv[0]时发生了浅拷贝,所以ret[0]中的vector< int >的_start、_finish、_end_of_storage和vv[0]的vector< int >的_start、_finish、_end_of_storage相同,即ret中的vecto< int >和vv中的vector< int >都两两指向了同一片空间,而vv的空间当出了generate函数就调用析构函数被销毁了,而ret中的vector< int >中的_start、_finish、_end_of_storage还指向了这些空间,所以才会出错。

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

我们需要重写vector的赋值运算符重载函数,将该函数写为深拷贝就可以解决问题了。
在实现vector的赋值运算符重载函数时,我们先实现swap函数,然后复用swap函数来完成赋值运算符重载函数。
当我们执行 v1 = v2时,赋值运算符重载函数的实参就为v2,而该函数为传值传参,所以当进入赋值运算符重载函数时,会调用拷贝构造函数创建一个临时对象v,而因为我们的拷贝构造函数使用new来申请空间,所以这个临时对象v被创建在堆区中,并且该临时对象v里的内容都和v2相同,这个临时对象v就相当于v2的副本,此时我们将这个临时对象v传入swap函数中,将v1的_start、_finish、_end_of_storage和这个临时对象v的_start、_finish、_end_of_storage进行交换,然后此时v1指向的_start、_finish、_end_of_storage中的内容和v2的内容相同,当执行完swap函数后将此时的v1返回。然后退出赋值运算符重载函数,因为临时对象v的作用域在赋值运算符重载函数中,所以退出赋值运算符重载函数时,临时对象v就会调用自己的析构函数将_start、_finish、_end_of_storage的内容释放,这样对象v1的原来的内存就被释放了。

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

相关文章:

c++-vector

文章目录 前言一、vector介绍二、vector使用1、构造函数2、vector 元素访问3、vector iterator 的使用4、vector 空间增长问题5、vector 增删查改6、理解vector<vector< int >>7、电话号码的字母组合练习题 三、模拟实现vector1、查看STL库源码中怎样实现的vector2…...

十四天学会C++之第二天(函数和库)

1. 函数的定义和调用 在C中&#xff0c;函数是组织和结构化代码的关键工具之一。它们允许您将一段代码封装成一个可重复使用的模块&#xff0c;这有助于提高代码的可读性和维护性。 为什么使用函数&#xff1f; 函数在编程中的作用不可小觑。它们有以下几个重要用途&#xf…...

蓝桥杯每日一题2023.10.3

杨辉三角形 - 蓝桥云课 (lanqiao.cn) 题目描述 题目分析 40分写法&#xff1a; 可以自己手动构造一个杨辉三角&#xff0c;然后进行循环&#xff0c;用cnt记录下循环数的个数&#xff0c;看哪个数与要找的数一样&#xff0c;输出cnt #include<bits/stdc.h> using na…...

JavaScript系列从入门到精通系列第十二篇:JavaScript中对象的简介和对象的基本操作以及JavaScript中的属性值和属性名

文章目录 前言 一&#xff1a;对象分类 1&#xff1a;内建对象 2&#xff1a;宿主对象 3&#xff1a;自建对象 二&#xff1a;对象的基本操作 1&#xff1a;创建对象 2&#xff1a;向对象中添加属性 3&#xff1a;读取对象中的属性 4&#xff1a;修改对象中的属性 三…...

OpenCV实现视频的追踪(meanshift、Camshift)

目录 1&#xff0c;meanshift 1.1 算法流程 1.2 算法实现 1.3 代码实现 1.4 结果展示 1&#xff0c;meanshift 1.1 算法流程 1.2 算法实现 1.3 代码实现 import numpy as np import cv2 as cv# 读取视频 cap cv.VideoCapture(video.mp4)# 检查视频是否成功打开 if n…...

并查集详解(原理+代码实现+应用)

文章目录 1. 并查集概念2. 并查集原理2.1 合并2.1 找根 3. 并查集实现3.1 结构定义3.2 FindRoot&#xff08;找根&#xff09;3.3 Union&#xff08;合并&#xff09;3.4 IsInSet&#xff08;判断两个值是否在一个集合里&#xff09;3.5 SetCount&#xff08;并查集中集合个数&…...

第k小的数

补充习题: 第k小的数 问题描述 有两个正整数数列,元素个数分别为 N N N和 M M M.从两个数列中分别任取一个数相乘,这样一共可以得到 N M N\times M NM个数,询问这 N M N\times M NM个数中第 K K K小的数是多少. 数据范围: N , M < 200000 , K < 2.1 ∗ 1 0 10 , …...

基于electron25+vite4创建多窗口|vue3+electron25新开模态窗体

在写这篇文章的时候&#xff0c;查看了下electron最新稳定版本由几天前24.4.0升级到了25了&#xff0c;不得不说electron团队迭代速度之快&#xff01; 前几天有分享一篇electron24整合vite4全家桶技术构建桌面端vue3应用示例程序。 https://www.cnblogs.com/xiaoyan2017/p/17…...

红米手机 导出 通讯录 到电脑保存

不要搞什么 云服务 不要安装什么 手机助手 不要安装 什么app 用 usb 线 连接 手机 和 电脑 手机上会跳出 提示 选择 仅传输文件 会出现下面的 一个 盘 进入 MIUI目录 然后进入 此电脑\Redmi Note 5\内部存储设备\MIUI\backup\AllBackup\20230927_043337 如何没有上面的文件&a…...

常见web信息泄露

一、源码(备份文件)泄露 1、git泄露 Git是一个开源的分布式版本控制系统&#xff0c;在执行git init初始化目录的时候&#xff0c;会在当前目录下自动创建一个.git目录&#xff0c;用来记录代码的变更记录等。发布代码的时候&#xff0c;如果没有把.git这个目录删除&#xff…...

找不到VCRUNTIME140_1.dll怎么办,VCRUNTIME140_1.dll丢失的5个解决方法

在当今的数字时代&#xff0c;我们的生活和工作都离不开电脑。然而&#xff0c;随着科技的发展&#xff0c;我们也会遇到各种各样的问题。其中&#xff0c;VCRUNTIME140_1.dll丢失的问题是许多人都会遇到的困扰。这个问题可能会导致许多应用程序无法正常运行&#xff0c;给我们…...

C#生成自定义海报

安装包 SixLabors.ImageSharp.Drawing 2.0 需要的字体&#xff1a;宋体和微软雅黑 商用的需要授权如果商业使用可以使用方正书宋、方正黑体&#xff0c;他们可以免费商用 方正官网 代码 using SixLabors.Fonts; using SixLabors.ImageSharp; using SixLabors.ImageSharp.Draw…...

BP神经网络的MATLAB实现(含源代码)

BP(back propagation)神经网络是1986年由Rumelhart和McClelland为首的科学家提出的概念&#xff0c;是一种按照误差逆向传播算法训练的多层前馈神经网络&#xff0c;是应用最广泛的神经网络模型之一 具体数学推导以及原理在本文不做详细介绍&#xff0c;本文将使用MATLAB进行B…...

AES和Rijndael的区别

快速链接: . 👉👉👉 个人博客笔记导读目录(全部) 👈👈👈 付费专栏-付费课程 【购买须知】:密码学实践强化训练–【目录】 👈👈👈“Rijndael” 这个词的中文谐音可以近似地发音为 “瑞恩达尔”。请注意,这只是一种近似的发音方式,因为该词是荷兰姓氏 “Ri…...

【数据结构】—堆详解(手把手带你用C语言实现)

食用指南&#xff1a;本文在有C基础的情况下食用更佳 &#x1f525;这就不得不推荐此专栏了&#xff1a;C语言 ♈️今日夜电波&#xff1a;水星—今泉愛夏 1:10 ━━━━━━️&#x1f49f;──────── 4:23 …...

关于算法复杂度的几张表

算法在改进今天的计算机与古代的计算机的区别 去除冗余 数据点 算法复杂度 傅里叶变换...

蓝桥杯每日一题2023.10.1

路径 - 蓝桥云课 (lanqiao.cn) 题目分析 求最短路问题&#xff0c;有多种解法&#xff0c;下面介绍两种蓝桥杯最常用到的两种解法 方法一 Floyd&#xff08;求任意两点之间的最短路&#xff09;注&#xff1a;不能有负权回路 初始化每个点到每个点的距离都为0x3f这样才能对…...

第三章:最新版零基础学习 PYTHON 教程(第十节 - Python 运算符—Python 中的运算符重载)

运算符重载意味着赋予超出其预定义操作含义的扩展含义。例如,运算符 + 用于添加两个整数以及连接两个字符串和合并两个列表。这是可以实现的,因为“+”运算符被 int 类和 str 类重载。您可能已经注意到,相同的内置运算符或函数对于不同类的对象显示不同的行为,这称为运算符…...

Nacos 实现服务平滑上下线(Ribbon 和 LB)

前言 不知道各位在使用 SpringCloud Gateway Nacos的时候有没有遇到过服务刚上线偶尔会出现一段时间的503 Service Unavailable&#xff0c;或者服务下线后&#xff0c;下线服务仍然被调用的问题。而以上问题都是由于Ribbon或者LoadBalancer的默认处理策略有关&#xff0c;其…...

c/c++里 对 共用体 union 的内存分配

对union 的内存分配&#xff0c;是按照最大的那个成员分配的。 谢谢...

博途SCL区间搜索指令(判断某个数属于某个区间)

S型速度曲线行车位置控制,停靠位置搜索功能会用到区间搜索指令,下面我们详细介绍区间搜索指令的相关应用。 S型加减速行车位置控制(支持点动和停车位置搜索)-CSDN博客S型加减速位置控制详细算法和应用场景介绍,请查看下面文章博客。本篇文章不再赘述,这里主要介绍点动动和…...

(三)激光线扫描-中心线提取

光条纹中心提取算法是决定线结构光三维重建精度以及光条纹轮廓定位准确性的重要因素。 1. 光条的高斯分布 激光线条和打手电筒一样,中间最亮,越像周围延申,光强越弱,这个规则符合高斯分布,如下图。 2. 传统光条纹中心提取算法 传统的光条纹中心提取算法有 灰度重心法、…...

递归与分治算法(1)--经典递归、分治问题

目录 一、递归问题 1、斐波那契数列 2、汉诺塔问题 3、全排列问题 4、整数划分问题 二、递归式求解 1、代入法 2、递归树法 3、主定理法 三、 分治问题 1、二分搜索 2、大整数乘法 一、递归问题 1、斐波那契数列 斐波那契数列不用过多介绍&#xff0c;斐波那契提出…...

Java之SpringCloud Alibaba【六】【Alibaba微服务分布式事务组件—Seata】

一、事务简介 事务(Transaction)是访问并可能更新数据库中各种数据项的一个程序执行单元(unit)。 在关系数据库中&#xff0c;一个事务由一组SQL语句组成。 事务应该具有4个属性: 原子性、一致性、隔离性、持久性。这四个属性通常称为ACID特性。 原子性(atomicity) ∶个事务…...

Android逆向学习(五)app进行动态调试

Android逆向学习&#xff08;五&#xff09;app进行动态调试 一、写在前面 非常抱歉鸽了那么久&#xff0c;前一段时间一直在忙&#xff0c;现在终于结束了&#xff0c;可以继续更新android逆向系列的&#xff0c;这个系列我会尽力做下去&#xff0c;然后如果可以的话我看看能…...

音频编辑软件Steinberg SpectraLayers Pro mac中文软件介绍

Steinberg SpectraLayers Pro mac是一款专业的音频编辑软件&#xff0c;旨在帮助音频专业人士进行精细的音频编辑和声音处理。它提供了强大的频谱编辑功能&#xff0c;可以对音频文件进行深入的频谱分析和编辑。 Steinberg SpectraLayers Pro mac软件特点 1. 频谱编辑&#xff…...

基于.Net Core实现自定义皮肤WidForm窗口

前言 今天一起来实现基于.Net Core、Windows Form实现自定义窗口皮肤&#xff0c;并实现窗口移动功能。 素材 准备素材&#xff1a;边框、标题栏、关闭按钮图标。 窗体设计 1、创建Window窗体项目 2、窗体设计 拖拉4个Panel控件&#xff0c;分别用于&#xff1a;标题栏、关…...

【Rust】操作日期与时间

目录 介绍 一、计算耗时 二、时间加减法 三、时区转换 四、年月日时分秒 五、时间格式化 介绍 Rust的时间操作主要用到chrono库&#xff0c;接下来我将简单选一些常用的操作进行介绍&#xff0c;如果想了解更多细节&#xff0c;请查看官方文档。 官方文档&#xff1a;chr…...

blender快捷键

1&#xff0c; shift a 添加物体 2&#xff0c;ctrl alt q 切换四格视图 3, ~ 展示物体的各个视图按钮&#xff0c;&#xff08;~ 就是tab键上面的键&#xff09; 4&#xff0c;a 全选&#xff0c;全选后&#xff0c;点 ctrl 鼠标框选 减去已经选择的&#xff1b…...

java Spring Boot 自动启动热部署 (别再改点东西就要重启啦)

上文 java Spring Boot 手动启动热部署 我们实现了一个手动热部署的代码 但其实很多人会觉得 这叫说明热开发呀 这么捞 写完还要手动去点一下 很不友好 其实我们开发人员肯定是希望重启这种事不需要自己手动去做 那么 当然可以 我们就让它自己去做 Build Project 这个操作 我们…...

什么是网站app/培训总结心得体会

一、今天学习到了vim编辑器的快捷键操作 全选复制&#xff1a;gg->y->G 全选删除&#xff1a;gg->d->G 二、shell编程方面 1、echo -e 表示开始转义&#xff0c;所以 echo -e "Hello \n World"执行结果是&#xff1a; Hello World 2、echo -n 表示…...

做网站一定要学java吗/网站推广计划书

为了方便读者朋友们自学桌面端编程&#xff0c;本文介绍了一些在线课程。桌面端编程方面的在线课程非常多&#xff0c;更新速度快&#xff0c;读者朋友们也可以在网上自行搜索&#xff0c;选择适合自己的在线课程进行学习。如需要打开课程的网站&#xff0c;请复制课程的网址到…...

企业内部系统网站制作/常州网站建设优化

計算机组成原理FPGA实验指导书《计算机组成原理》实验指导书计算机科学与技术学院目录实验一 熟悉实验平台……………………………………………………………………(3)实验二 总线传送…………………………………………………………………………(10)实验三 运算器的设计与调试……...

品牌网站建设怎么样/优秀网站

织梦内容管理系统(DedeCms) 以简单、实用、开源而闻名&#xff0c;是国内最知名的PHP开源网站管理系统&#xff0c;也是使用用户最多的PHP类CMS系统&#xff0c;在经历多年的发展&#xff0c;目前的版本无论在功能&#xff0c;还是在易用性方面&#xff0c;都有了长足的发展和进…...

北京顺义做网站/互联网运营培训课程

我编写了几个程序,我编写了一些程序,但知道我编写了一个Chat messenger.当我编译服务器或客户端时,我总是从javac得到错误. “错误&#xff1a;找不到符号”.并且错误是在其他类的构造函数中应该如何构造那里.例&#xff1a;Chat.java:11 error: cannot find symbolFrame frm …...

枫叶主机 wordpress/谷歌seo建站

准备我测试使用的Python版本为2.7.10&#xff0c;如果你的版本是Python3.5的话&#xff0c;这里就不太适合了。使用Speech API原理我们的想法是借助微软的语音接口&#xff0c;所以我们肯定是要进行调用 相关的接口。所以我们需要安装pywin32来帮助我们完成这一个底层的交互。示…...