【C++】STL——vector 深度剖析 及 模拟实现(二)

举报
YIN_尹 发表于 2023/09/22 22:15:34 2023/09/22
【摘要】 2.4 迭代器失效问题会引起其底层空间改变的操作会引起其底层空间改变的操作,都有可能导致迭代器失效,比如:resize、reserve、insert、assign、push_back等。出错原因:以上操作,都有可能会导致vector扩容,也就是说vector底层原理旧空间被释放掉,而在打印时,it还使用的是释放之间的旧空间,在对it迭代器操作时,实际操作的是一块已经被释放的空间,而引起代码运...

2.4 迭代器失效问题

会引起其底层空间改变的操作

会引起其底层空间改变的操作,都有可能导致迭代器失效,比如:resize、reserve、insert、assign、push_back等。

出错原因:

以上操作,都有可能会导致vector扩容,也就是说vector底层原理旧空间被释放掉,而在打印时,it还使用的是释放之间的旧空间,在对it迭代器操作时,实际操作的是一块已经被释放的空间,而引起代码运行时崩溃。

解决方式:

在以上操作完成之后,如果想要继续通过迭代器操作vector中的元素,可以更新it。


首先我们先来实现一下insert:

e855f028ca0c4e44a530c1c7dda139d9.png

怎么实现?

逻辑是不是很简单啊,就是挪动数据,然后填值嘛。

首先应该判断一下pos是否合法以及是否需要扩容:

eb683e50915b462590c50c098172dc19.png

这就写好了。

来,测试一下:

143e42474d024694b8df2bbbb306d2ee.png

嗯???

怎么回事?

第一次insert还正常,第二次怎么出来个个随机值啊。

🆗,通过调式和分析我们会发现原因在于第二次insert的时候发生了扩容,那为什么发生了扩容就出问题了呢?

889dae24227e47ec940957751ff43d9d.png

我们这里扩容是不是异地开一块新空间,然后拷贝数据,释放旧空间啊,那这样的话,扩容之后,_start和_finish是不是就变了啊,现在它们指向一块新空间,而pos呢,是不是还是原来的pos

0f024e9460fe4b53aa1a1c81bd449f7c.png

那现在pos和end的大小关系是不是未知的啊,这里循环会走多少次我们也不知道。

此时pos指向的空间以及被释放了,即pos此时是一个野指针了,但我们又把val放到了pos指向的空间,但是对扩容后的新空间并没有影响,只是把_finish++了一次,所以打印出来是随机值。

🆗,那这里其实就是迭代器失效的一种情况,扩容之后pos位置的这个迭代器就失效了。

怎么解决?

是不是如果扩容的话,我们得去更新一下pos啊

ededebf3ced24ab5a5166b4213dda312.png

然后我们再来测试一下:

2ddcd5613c054869880f84345b101cdd.png

🆗,就没问题了。

当然这只是迭代器失效中的一种情况,后面可能还会有一些更复杂的场景,等着我们解决。

再来看一个问题:

d1677db8346445f49983e95661144dde.png

我们现在通过find获取元素3的位置pos,然后在该位置insert一个新元素。

那我们现在想去修改pos位置的这个值。可以吗?

f3150608eafb43bb80488e02f7e84cfc.png

我们find的是3的位置,但是后面++却++的不是3,为什么?

因为我们又insert了一个值到pos位置,3是不是就被挪到后面了啊。

另外如果是这种情况:

23820e3557d84a4bad1cf56554fad1a9.png

哦豁,这次++怎么没起作用啊,谁都没有++。

ccfff1781e074192974ce45e200b5473.png

pos里面怎么放了一个随机值啊。

来看我们这次insert的时候是不是扩容了啊,初始我们给了4个空间嘛。

但是,我们刚才不是已经处理了扩容时迭代器失效的问题,更新了pos嘛,为什么还有问题。

🆗,我们确实更新了,但我们更新的是不是insert函数里面的pos啊,而且我们是传值,所以并没有影响外面的pos,外面的pos在扩容之后还是野指针。

所以外面的pos还是会失效,我们(*pos)++没有对任何元素产生影响。


所以呢:


刚才上面这两种情况,第一次虽然insert之后pos还能用,好像没失效(因为没扩容),但是可能已经不是我们想要的位置了,就像上面我们find的是3的位置,但后面++的并不是3 了。

第二种情况由于发生了扩容,pos这个位置的迭代器就是一个野指针了,彻底失效了。

但是,这两种情况,我们都认为是迭代器失效。


那上面第二个情况怎么解决啊?


传值不行,传引用吗?

cb42b14d22d7433ea4d9cc6af5224955.png

这样外面的pos确实不是野指针了,但是,这样又不行了:

af3905eb723f4a9d909a390cb863c1a5.png

为什么?

f183c12cb8a045faa3e46bc3bddfcb8f.png

这里是传值返回,返回的是拷贝的临时变量,具有常性,不能传给引用。

那怎么办?7af79d4fc9be49f59220e28199398e6d.png

我们看到库里面的insert是有返回值的,返回更新后的pos:

dadc530db6c747e3a9cf941da50850f8.png

08d85c0f2e22433b8f141e52d4064212.png


6db22a3a870d43e7adca80f139ba73c6.png

接收一下返回值,就可以了。

所以:

insert之后我们就认为迭代器失效了,就不要使用了。

我们可以重新去find,或者根据需要接收一下返回值。

指定位置元素的删除操作–erase

那我们再来实现一下erase:

faf55cb232f54c16a63c418da557904d.png

删除pos位置的元素,那我们就直接挪动后面的元素覆盖就行了,如果后一个的话直接- -_finish就行了。

cde7a3b527ca46bfbd2ebb662bab9e9b.png

试一下:

37f404301fce452085fa41d4e67fab22.png

那大家思考一下,erase之后迭代器是否失效?

我们试一下:

dc62b0dc0e5a46b4841105cbd2585773.png

erase删除pos位置元素后,pos位置之后的元素会往前搬移,没有导致底层空间的改变,理论上讲迭代器不应该会失效,但是:如果pos刚好是最后一个元素,删完之后pos刚好是end的位置,而end位置是没有元素的,那么pos就失效了。因此删除vector中任意位置上元素时,vs就认为该位置迭代器失效了。 并且进行了强制检查。

我们看一下vs 上std中的处理:

2d6f31f2871c4456acfcb2eef5dbd3a0.png

我们调式会发现

4ad16686d74d4744ae1cb9496d4a2369.png

在这里挂了。

删除vector中任意位置上元素时,vs就认为该位置迭代器失效了

注意⚠:Linux下,g++编译器对迭代器失效的检测并不是非常严格,处理也没有vs下极端。

上面那段代码如果我们在G++环境下面运行会发现结果跟我们自己模拟实现的是一样的,因为我们就是按照SGI版本实现的嘛,G++就是采用的SGI,底层跟我们一样是原生指针。

SGI STL中,erase导致迭代器失效后,程序不一定会崩溃,但运行结果可能是不对的;如果迭代器位置不在begin和end范围内,肯定会崩溃的。


所以:


erase以后,我们也认为迭代器失效了,就不要再使用了,行为结果未定义(不同平台可能不同)。

当然库里面的实现也是有返回值的:

84f1c5380c954288a1ade60caeffd05d.png

所以我们可以根据需要接收返回值去更新失效的迭代器。

c92b20665c8e48dbb98d88fc86740f23.png

然后大家思考一下,我们之前学的string存在迭代器失效的问题吗?

当然也是存在的,但是我们为什么讲string的时候并没有提迭代器失效的问题呢?

🆗,因为string的insert和erase是不是提供了下标的版本啊

d556ebe7b45b4bbcbae79e446491f4e1.png

5a091f3acf1946f6b7a777b065d79ae8.png

用下标的版本是不是就不存在迭代器失效的版本啊,但是它也提供了迭代器的版本,那当然如果用迭代器就跟vector一样了,要考虑迭代器失效的问题。

2.5 剩余接口补充

2.5.1 析构

接下来我们来实现一下析构:

vector涉及资源申请,析构函数是需要我们自己实现的:

985801c1f2d24d90b2ec34c618fe03d0.png

2.5.2 其它版本的构造

构造函数其实我们一开始就实现了,不过我们只实现了一个默认构造:

e6113be5b82b40f4ab46960f27622e46.png

那接下来我们再来实现几个其它版本的:

2e627477b6c94db4992604b1d1ce5948.png

首先看这个,n个val去初始化对象:

f671db3f1f8146bc9e1304dd5617218b.png

首先这里val的缺省值是T(),相信这个不用给大家解释了(调用默认构造函数产生一个匿名对象去作为这个缺省值),我们上面刚刚讲解过,因为这里是模板,要针对所有类型。

那大家看,这里是用产生的这个匿名对象的引用去初始化对象,但是我们之前学了匿名对象的声明周期不是只在当前这一行吗?

那还怎么拿它去初始化呢?

🆗,我们在之前讲解匿名对象的时候也提了:

和临时变量一样,如果我们用匿名对象去初始化一个引用的话,它的生命周期就会被延长至该引用被销毁。并且这里肯定都要加const的,因为临时变量和匿名对象都具有常性。


那函数内部具体怎么实现呢?


🆗,我们是不是可以直接push_back() n次啊,那这样的话我们还可以使用reverse直接把空间开够,避免push_back过程中扩容:

02ee7ef266054e26b35352bb7c752123.png

来测试一下:

3fea2620b795403c9a1d7bcce08f2a8f.png

没有问题。

然后我们再来看一下这个构造函数:

0dae4688473d4a58a1e713e27750aa24.png

用一段迭代器区间去初始化,它的使用我们在上面已经讲解过了,那现在我们来尝试模拟实现一下它:

我们看到它其实是一个函数模板,因为它还支持接收除vector之外其它类型的迭代器,这个我们上面也说过了。
所以说,在一个类模板中,它的成员函数可以是函数模板。

4e65cadfbf594e3299b0d18a46cde291.png那这里的模板参数我们可以用T,也可以直接用这个InputIterator,因为模板参数的名字我们可以自己起嘛。

那函数体内怎么实现呢?

也很简单,我们直接遍历这个迭代器区间一个个push_back()就行了:

0975712247774f79a47b6213da2a8608.png

当然这个地方如果大家觉得每个构造函数都要写初始化列表,嫌麻烦的话,我们之前学过C++11不是支持类中的非静态成员变量可以给缺省值嘛,所以我们也可以直接给缺省值,这样就不用写初始化列表了。

bea79659168a4492b4a7fe080320a865.png

🆗,我们接着看:

迭代器区间的这个构造函数我们写好之后呢,会发现一个问题:


a99198ec2ca74b6b8b5ea032f4766ed0.png

我们编译会发现报错了,说非法的间接寻址。

哎?怎么回事啊?

329fb18c7ecb4a4db13ab09a25679914.png

然后我们双击下面这条错误信息发现定位到了这里

dfdb82168334465394c7f19ad2ea97cf.png

把它注释掉就没事了,说明问题与这段代码有关系。

那原因是什么呢?

f2abe49357d242adafc4c1cae41b5f86.png

其实根据错误信息我们也能看出来,yin::vector a(10, 5);这句代码匹配到了迭代器区间的这个构造函数,但是我们本意是不是让他去匹配n个val的构造函数啊。

那这里为什么会匹配到迭代器区间的这个?

🆗,因为函数调用根据参数去匹配的时候,会去找最合适,最匹配的那个,刚才我们没实现迭代器的这个版本的时候,那它只有一个选择,所以就没事,但是现在实现之后,那他就选择了迭代器区间这个,因为这个是一个函数模板,这里10跟5两个int传过来,模板参数InputIterator就自动被推演成int类型了。

但是这个的话:

e74c611379624257a7da92603a681a3c.png

n是size_t 类型的,int过了还要进行一个类型转换。

所以就匹配到了模板的那个版本。

66eb3cc51beb45c2a083a6e68344b22f.png

这里我们加个u就不报错了,加个u它就是无符号整型了嘛。

那这里我们怎么解决这个问题呢?

🆗 ,我们可以看一下源码是怎么做的:

1d041602930c4ecdbed1febbfc30111c.png

我们看到源码里是不是重载了多个版本来解决参数类型的这个问题啊。

当然我们这里不用考虑那么多情况:

6f4b9c1d876b47038326edb6c131627f.png

032a4cf4fa7c4ba2990ebd7b94b5a317.png

是不是就可以了。

那我们来测试一下迭代器区间的这个构造函数:

530b1357b5664417b52bd1717149a1d3.png

当然我们可以自己控制这个区间的大小。

967acce7fbbf495dbb6c53106fbddf4f.png

当然我们也可以使用其它类型的迭代器

这个我们再讲使用的时候也演示过,那我们自己写的这个当然也是可以的:

b1ea724c093841759659f6a8baf5fc88.png

那除此之外,还能这样玩:

28f70e776bca47868260131b3429393e.png

甚至可以是数组,我相信原因不用过多解释了:

883b621896e54ae0a4617b8ea1514de1.png


2.5.3 对vector排序

我们现在相对一个vector进行排序,怎么搞?

6729b7c4c82d4b539892ed441da392c3.png

那其实算法库里面提供了一个排序的函数sort,我们可以直接用。

我们看到它也是接收一段迭代器区间,演示一下:

a60ce5bd433f4e1fab8e3b6048bd5374.png

就排好了。

当然也可以直接排数组:

cf1a16c7fd764484b0bdbda398e9b159.png

当然我们也看出来了它默认是升序。

如果想降序的话这样就行了:

112a3f60452f4950a88fbcce5266bdd2.png

这个大家先简单了解一下,后面还会讲。

2.6 使用memcpy拷贝的问题

我们现在还没有实现拷贝构造:

拷贝构造如果我们不写,编译器会默认生成,但是默认生成的是浅拷贝,而我们的vector涉及资源申请,所以如果用默认生成的一定是会出问题的:

12c5af0cab2d4aaaa8c7f040c77d38c2.png

相信原因就不用给大家过多解释了。

所以我们要自己实现一下拷贝构造:

那有了我们之前实现的接口,其实要再来实现这个拷贝构造,也很简单:

92af799923394be1b8e6aa19cc7f5c01.png

怎么做?

🆗,直接使用reverse把空间开好,然后遍历v的所有元素push_back到当前对象是不是就行了:

007793d5ff5848f5a611f3e9824caddb.png

试一下:

17d3a9454a554c96bd13d3022dad427a.png

这下是不是就行了啊。

但是呢:

上面这种写法不够传统,本质是在复用,接下来我们再实现一个更传统一点的,方便我们讲解后面的东西:a10c2a4e2867432abeb1fffee20a39a2.png

大家思考一下,如果不复用push_back的话,开好空间之后我们可以怎么拷贝?

a930fd7557824f16af9ae7e8f843fad4.png

是不是可以用memcpy啊:

90ca65e5b655473581ea8e215ff9474b.png

1013d1963233487db762f7d91ada5c29.png🆗,可以完成。

但是,如果是这种场景呢:

167bc07dd8854ded8a56cec0538c8a42.png

刚才vector的元素类型是int,如果是string呢?

我们来运行程序:

1d10b115f8664811b1674b6ede248c44.png

好像挂了呀,调式一下:

c99a33119b7d4de3b7e3c4eb0d6fc5bb.png

确实是挂了。


怎么回事呢?为什么刚才vector就没事,而vector就出现问题了呢?


🆗,原因就在于我们拷贝构造内部使用了memcpy来拷贝数据。

1. memcpy是内存的二进制格式拷贝,将一段内存空间中内容原封不动的拷贝到另外一段内存空间中

2. 如果拷贝的是自定义类型的元素,memcpy既高效又不会出错

7260424320b540db87442314f0fadd00.png

但如果拷贝的是自定义类型元素并且
自定义类型元素中涉及到资源管理时,就会出错,因为memcpy的拷贝实际是浅拷贝

ea64fa62ae7e4ae0a0b3cc4b93cd1aaa.png

那这样在析构的时候是不是又会出现对同一块空间析构两次的情况啊,所以才会导致程序崩溃。


所以:


vector里面嵌int这些类型的数据(vector)是没问题的,但如果嵌的是自定义类型并且该自定义类型设计资源申请(需要深拷贝,比如vector,vector等),就会出错,因为内部memcpy的拷贝实际是浅拷贝。


那我们可以怎么解决这个问题呢?


也好处理,可以这样搞:

0e8185814d6e484a8bb80b761a4b93c3.png

大家看这样的话,如果int这样的内置类型就直接赋值,这肯定是没问题的,如果是涉及资源管理的自定义类型,是不是就去调深拷贝的拷贝构造就行了啊。

测试一下:

d03e633957534d33acde4d4b214ee4e4.png

就可以了。

2.7 对vector的理解

vector的使用(OJ练习)

首先我们来看一道题:

题目链接: link

bb6cbe45a8634ebc9980ed061def37d9.png

思路讲解

这道题呢是给我们一个行数 numRows,生成「杨辉三角」的前 numRows 行。

那这道题的思路是很简单的,没什么难度:

核心思想:找出杨辉三角的规律,发现每一行头尾都是1,中间第[j]个数等于上一行[j-1]+[j]。

但是,如果我们用C语言去写这道题:

9e96fa66b95a4451a4184abc926d98b9.png

大家看,其实是有点麻烦的,一级指针、二级指针,最终返回的数组还得是malloc的。首先这个参数可能就给我们看懵逼了。

而C++呢?

31b3870d9c13454c808bdfbaf3af6d3c.png

C++有了vector,就爽很多了。

不过这里需要用到vector,这是什么东西,🆗,就是一个二维的vector嘛,类似于二维数组,但是如果是二维数组,我们开一个m x n的,它的每行元素是不是相等的啊,而这里杨辉三角是不是每行元素个数不一样啊。

但是用vector,我们是不是就可以很方便的控制每行的size啊。

fb9ff8d91ba74b74b2c785f3eeeea294.png

我们只需要定义一个numRows行的vector,每行的元素个数是多少?

是不是i+1啊,i等于0(第0行),就是一个元素;i等于1,两个元素;以此类推。

然后只需把每行第一个和最后一个元素初始化为1 ,中间剩余的元素是不是就是它上一行的元素和上一行的前一个元素相加的和啊。

AC代码

class Solution {
public:
    vector<vector<int>> generate(int numRows) {
        vector<vector<int>> vv(numRows);
        for(int i=0;i<numRows;i++)
        {
            vv[i].resize(i+1,0);
            vv[i][0]=vv[i][vv[i].size()-1]=1;
        }
        for(size_t i=0;i<vv.size();++i)
        {
            for(size_t j=0;j<vv[i].size();++j)
            {
                if(vv[i][j]==0)
                    vv[i][j]=vv[i-1][j]+vv[i-1][j-1];
            }
        }
        return vv;
    }
};

8fc477a6743d4320a5c65e09bc796794.png

构造一个vv动态二维数组,vv中总共有n个元素,每个元素都是vector类型的,每行没有包含任何元素,如果n为5时如下所示:

b7f90d161f614000a27689e2c05bdc97.png

vv中元素填充完成之后,如下图所示:

d7d02f778001404a8148a96d7ffe27f4.png

vector的拷贝构造

我们上面已经实现了vector的拷贝构造,并且解决了使用memcpy拷贝的内部浅拷贝的问题。


vector的拷贝构造


首先经过我们上面的修改,我们的拷贝构造应该是没问题了,上面我们测试vector的拷贝构造就没问题了。

我们就来拷贝构造一下上面的那个杨辉三角,用我们自己实现的vector:

4d18c89da32e4d448e9d5f568a541c5e.png

哎呀!怎么回事?

我们的拷贝构造上面不是已经修改好了嘛,为什么又出错了。

问题出在哪里呢?

🆗,除了我们的拷贝构造之前用了memcpy,不过我们已经改了,可是,还有一个地方我们也用了memcpy:

8f8528ae9c5044afb6893c2ebddc7863.png

我们的reverse也用了memcpy拷贝数据。

>那这个地方是不是也会出现浅拷贝的问题对于vector这些类型来说。

把里面的每个小vector浅拷贝到新空间,但是两块空间里面的每个小vector指向同一块空间,然后接着就直接把旧空间释放掉了,然后扩容之后新空间里面的每个vector的_start就全是野指针了。

那要怎么修改?

和拷贝构造一样,我们可以一个一个赋值:

11b6eff4b4a742109743b85c32cce780.png

然后我们再来试一下:

849e162cf90c4ff6b1b7d600dbbbbe26.png

哎呀哈?怎么还有问题?

🆗,那我们通过调式来尝试找一下问题:

1831edc9f97e4c8f9457dbe7d7823529.png

我们调式一步步走会发现走到析构的时候崩了。

但是我们的拷贝构造不是应该没问题了嘛,我们来分别观察一下函数返回的ret和main函数中的vv:

86ac62129071474ead390b1d1ac1006d.png

8e364b27694b4b13ae54f8de9978910b.png

我们仔细观察一下能够看出来,ret和vv的地址是不一样的,这当然是正常的表现;但是它们两个里面每个小vector的地址却是一样的,这就有问题了呀!

这说明什么?

内部小vector的拷贝是不是又是浅拷贝啊,欸!可是,我们不是已经修改过拷贝构造了嘛,而且上面vector测试也没问题啊。


请问问题出在哪里?


🆗,我们拷贝构造现在是怎么实现的?

51eb8b8ab5a04c1bbcfdbc45edbb1280.png

由于memcpy会导致内层浅拷贝的问题,所以我们改成了一个元素一个元素去赋值,内置类型之间赋值没有问题,涉及资源申请的自定义类型赋值会调深拷贝的赋值重载,我们拷贝构造的实现没什么问题。

但是:

问题就在于我们现在使用的vector是我们自己写的,而现在我们并没有进行赋值重载,而默认生成的是浅拷贝,所以:

才导致了内层小vector的赋值是浅拷贝。


所以,我们现在必须实现深拷贝的赋值重载:


怎么搞呢?

很简单,比如v1=v2,我们可以直接这样:

427a71e528d04d91a95f82485e255454.png

比如要v2赋值给v1,那赋值重载函数中的参数v是v2的拷贝,然后直接把v交换给v1就行了。当然这里我们再实现一个交换的接口。

那这下我们再来测试一下:

3a137925dc2a45d09cb769677df8f8dc.png

就可以了。

另外:

我们说了,上面实现的那个拷贝构造是比较传统的写法,那现在我们还可以写一个现代的写法:

133c14737e844cd8bfcce044a40815be.png

直接借助迭代器版本的拷贝构造构造出来一个对象,然后把它交换给目标对象。

0f191ec6a44c48f5913e1b71e282dd99.png

也是没问题的。

然后再给大家提一个东西就是:

f846da1a4e314bda807bbed5fc679bbd.png

这个地方不加模板参数语法也是支持的,但是建议大家自己写的时候还是加上,给大家说一下的目的是如果我们在别的地方看到这个的写法也不要奇怪,知道这样也可以。

3. 源码展示

vector.h

#pragma once
#include <assert.h>
#include <string.h>

namespace yin
{
    template<class T>
    class vector
    {
    public:
        typedef T* iterator;
        typedef const T* const_iterator;

        vector()
        {}
        vector(size_t n, const T& val = T())
        {
            reverse(n);
            for (size_t i = 0; i < n; ++i)
            {
                push_back(val);
            }
        }
        vector(int n, const T& val = T())
        {
            reverse(n);
            for (int i = 0; i < n; ++i)
            {
                push_back(val);
            }
        }
        template <class InputIterator>
        vector(InputIterator first, InputIterator last)
        {
            while (first != last)
            {
                push_back(*first);
                ++first;
            }
        }
        ~vector()
        {
            delete[] _start;
            _start = _finish = _end_of_storage = nullptr;
        }
        size_t capacity() const
        {
            return _end_of_storage - _start;
        }
        size_t size() const
        {
            return _finish - _start;
        }
        iterator begin()
        {
            return _start;
        }
        iterator end()
        {
            return _finish;
        }
        const_iterator begin()const
        {
            return _start;
        }
        const_iterator end()const
        {
            return _finish;
        }
        void reverse(size_t n)
        {
            if (n > capacity())
            {
                size_t sz = size();
                T* tmp = new T[n];
                if (_start)
                {
                    //memcpy(tmp, _start, size() * sizeof(T));
                    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())
        {
            if (n < size())
            {
                _finish = _start + n;
            }
            else
            {
                if (n > capacity())
                {
                    reverse(n);
                }
                //初始化
                while (_finish != _start + n)
                {
                    *_finish = val; 
                    ++_finish;
                }
            }
        }
        /*vector(const vector<T>& v)
        {
            reverse(v.capacity());
            for (auto e : v)
            {
                push_back(e);
            }
        }*/
        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)
        {
            vector<T> tmp(v.begin(), v.end());
            swap(tmp);
        }
        //vector(const vector<T>& v)
        //{
        //  _start = new T[v.capacity()];
        //  //memcpy(_start, v._start, sizeof(T) * v.size());
        //  for (size_t i = 0; i < v.size(); ++i)
        //  {
        //      _start[i] = v._start[i];
        //  }
        //  _finish = _start + v.size();
        //  _end_of_storage = _start + v.capacity();
        //}
        iterator insert(iterator pos, const T& val)
        {
            assert(pos >= _start && pos <= _finish);
            if (_finish == _end_of_storage)
            {
                size_t len = pos - _start;
                reverse(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 && pos < _finish);
            iterator start = pos + 1;
            while (start != _finish)
            {
                *(start - 1) = *start;
                ++start;
            }
            --_finish;
            return pos;
        }
        void push_back(const T& x)
        {
            if (_finish == _end_of_storage)
            {
                //扩容
                reverse(capacity() == 0 ? 4 : capacity() * 2);
            }
            *_finish = x;
            ++_finish;
        }
        void pop_back()
        {
            assert(!empty());
            --_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];
        }
        bool empty()
        {
            return _start == _finish;
        }
    private:
        iterator _start = nullptr;
        iterator _finish = nullptr;
        iterator _end_of_storage = nullptr;
    };
}

test.cpp

#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
using namespace std;
#include "vector.h"
//#include <vector>
//int main()
//{
//  vector<int> a;
//  a.push_back(1);
//  a.push_back(2);
//  a.push_back(3);
//  a.push_back(4);
//  a.push_back(5);
//
//  for(size_t i = 0; i < a.size(); ++i)
//  {
//      cout << a[i] << " ";
//  }
//  cout << endl;
//  for (vector<int>::iterator it = a.begin(); it != a.end(); ++it)
//  {
//      cout << *it << " ";
//  }
//  cout << endl;
//  for (auto e : a)
//  {
//      cout << e << " ";
//  }
//  cout << endl;
//  return 0;
//}

//int main()
//{
//  vector<int> a;
//  a.push_back(1);
//  a.push_back(2);
//  a.push_back(3);
//  a.push_back(4);
//  a.push_back(5);
//
//  for (vector<int>::iterator it = a.begin(); it != a.end(); ++it)
//  {
//      cout << *it << " ";
//  }
//  cout << endl;
//
//  for (vector<int>::reverse_iterator rit = a.rbegin(); rit != a.rend(); ++rit)
//  {
//      cout << *rit << " ";
//  }
//  cout << endl;
//  return 0;
//}

//int main()
//{
//  vector<int> a;
//  a.push_back(1);
//  a.push_back(2);
//  a.push_back(3);
//  a.push_back(4);
//  a.push_back(5);
//  for (auto e : a)
//  {
//      cout << e << " ";
//  }
//  cout << endl;
//  vector<int>::iterator pos = find(a.begin(), a.end(), 3);
//  if (pos != a.end())
//  {
//      a.insert(pos, 30);
//  }
//  for (auto e : a)
//  {
//      cout << e << " ";
//  }
//  cout << endl;
//  pos = find(a.begin(), a.end(), 30);
//  if (pos != a.end())
//  {
//      a.erase(pos);
//  }
//  for (auto e : a)
//  {
//      cout << e << " ";
//  }
//  cout << endl;
//  a.erase(a.begin());
//  for (auto e : a)
//  {
//      cout << e << " ";
//  }
//  cout << endl;
//  return 0;
//}
//void func(const yin::vector<int>& a)
//{
//  for (size_t i = 0; i < a.size(); ++i)
//  {
//      cout << a[i] << " ";
//  }
//  cout << endl;
//  for (yin::vector<int>::const_iterator it = a.begin(); it != a.end(); ++it)
//  {
//      cout << *it << " ";
//  }
//  cout << endl;
//}
//int main()
//{
//  yin::vector<int> a;
//  a.push_back(1);
//  a.push_back(2);
//  a.push_back(3);
//  a.push_back(4);
//  a.push_back(5);
//  a.push_back(6);
//  func(a);
//  for (size_t i = 0; i < a.size(); ++i)
//  {
//      cout << a[i] << " ";
//  }
//  cout << endl;
//  a.pop_back();
//  a.pop_back();
//  a.pop_back();
//  for (yin::vector<int>::iterator it=a.begin(); it != a.end(); ++it)
//  {
//      cout << *it << " ";
//  }
//  cout << endl;
//  a.pop_back();
//  a.pop_back();
//  //a.pop_back();
//  //a.pop_back();
//  //a.pop_back();
//  //a.pop_back();
//  //a.pop_back();
//  //a.pop_back();
//  //a.pop_back();
//  //a.pop_back();
//  //a.pop_back();
//  for (auto e : a)
//      cout << e << " ";
//  cout << endl;
//  return 0;
//}
//#include <vector>
//int main()
//{
//  yin::vector<int> a;
//  a.push_back(1);
//  a.push_back(2);
//  a.push_back(3);
//  a.push_back(4);
//  for (auto e : a)
//      cout << e << " ";
//  cout << endl;
//  auto pos = find(a.begin(), a.end(), 3);
//  if (pos != a.end())
//  {
//      pos = a.erase(pos);
//  }
//  //(*pos)++;
//  for (auto e : a)
//      cout << e << " ";
//  cout << endl;
//  return 0;
//}

//#include <algorithm>
//int main()
//{
//  /*int arr[] = { 2,8,9,3,5,6,8,7 };
//  yin::vector<int> b(arr, arr + sizeof(arr) / sizeof(arr[0]));
//  for (auto e : b)
//      cout << e << " ";
//  cout << endl;
//  sort(b.begin(), b.end());
//  for (auto e : b)
//      cout << e << " ";
//  cout << endl;*/
//
//  int arr[] = { 2,8,9,3,5,6,8,7 };
//  for (auto e : arr)
//      cout << e << " ";
//  cout << endl;
//  sort(arr, arr + sizeof(arr) / sizeof(arr[0]), greater<int>());
//  for (auto e : arr)
//      cout << e << " ";
//  cout << endl;
//
//  return 0;
//}

//int main()
//{
//  yin::vector<string> a(5, "hello");
//  for (auto e : a)
//      cout << e << " ";
//  cout << endl;
//
//  yin::vector<string> b(a);
//  for (auto e : b)
//      cout << e << " ";
//  cout << endl;
//  return 0;
//}

yin::vector<yin::vector<int>> generate(int numRows) {
    yin::vector<yin::vector<int>> ret(numRows);
    for (int i = 0; i < numRows; i++)
    {
        ret[i].resize(i + 1, 0);
        ret[i][0] = ret[i][ret[i].size() - 1] = 1;
    }
    for (size_t i = 0; i < ret.size(); ++i)
    {
        for (size_t j = 0; j < ret[i].size(); ++j)
        {
            if (ret[i][j] == 0)
                ret[i][j] = ret[i - 1][j] + ret[i - 1][j - 1];
        }
    }
    return ret;
}
int main()
{
    yin::vector<yin::vector<int>> vv(generate(5));
    for (size_t i = 0; i < vv.size(); ++i)
    {
        for (size_t j = 0; j < vv[i].size(); ++j)
        {
            cout << vv[i][j] << " ";
        }
        cout << endl;
    }
    cout << endl;

    return 0;
}

🆗,那我们关于vector的讲解就差不多到这里了,欢迎大家指正!!!

a6f50088beee475d84fd734b4ab4d1a1.png

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。