网站建设报价表下载,网站建设培训中心,专门做电商的招聘网站,静态网站标准库类型vector表示对象的集合#xff0c;其中所有对象的类型都相同。集合中的每个对象都有一个与之对应的索引#xff0c;索引用于访问对象。因为vector容纳着”其他对象#xff0c;所以它也常被称作容器(container).第 II部将对容器进行更为详细的介绍。 要想使用…标准库类型vector表示对象的集合其中所有对象的类型都相同。集合中的每个对象都有一个与之对应的索引索引用于访问对象。因为vector容纳着”其他对象所以它也常被称作容器(container).第 II部将对容器进行更为详细的介绍。 要想使用vector,必须包含适当的头文件。在后续的例子中都将假定做了如下using声明#include vectorusing std::vector;C语言既有类模板(classtemplate),也有函数模板其中vector是一个类模板。只有对C有了相当深入的理解才能写出模板事实上我们直到第16章才会学习如何自定义模板。幸运的是即使还不会创建模板我们也可以先试着用用它。模板本身不是类或函数相反可以将模板看作为编译器生成类或函数编写的一份说明。编译器根据模板创建类或函数的过程称为实例化(instantiation),当使用模板时需要指出编译器应把类或函数实例化成何种类型。对于类模板来说我们通过提供一些额外信息来指定模板到底实例化成什么样的类需要提供哪些信息由模板决定。提供信息的方式总是这样即在模板名字后面跟一对尖括号在括号内放上信息。vector能容纳绝大多数类型的对象作为其元素但是因为引用不是对象(参见2.3.1节第45页)所以不存在包含引用的vectoro除此之外其他大多数(非引用)内置类型和类类型都可以构成vector对象甚至组成vector的元素也可以是vector。需要指出的是在早期版本的C标准中如果vector的元素还是vector(或者其他模板类型)则其定义的形式与现在的C11新标准略有不同。过去必须在外层vector对象的右尖括号和其元素类型之间添加一个空格如应该写成vectorvectorint 而非vectorvectorint现在大可不必
3.3.1定义和初始化vector对象
和任何一种类类型一样vector模板控制着定义和初始化向量的方法。表3.4列出了定义vector对象的常用方法。可以默认初始化vector对象(参见2.2.1节第 40页)从而创建一个指定类型的空 vectorvectorstring svec; //默认初始化svec不含任何元素 最好使用空的花括号进行默认初始化 vectorstring svec{};看起来空vector好像没什么用但是很快我们就会知道程序在运行时可以很高效地往 vector对象中添加元素。事实上最常见的方式就是先定义一个空vector,然后当运行时获取到元素的值后再逐一添加。当然也可以在定义vector对象时指定元素的初始值。例如允许把一个vector对 象的元素拷贝给另外一个vector对象。此时新 vector对象的元素就是原vector 对象对应元素的副本。注意两个vector对象的类型必须相同vectorint ivec; / / 初始状态为空/ / 在此处给ivec添加一些值vectorint ivec2 (ivec) ; // 把 ivec 的元素拷贝给 ivec2vectorint ivec3 ivec; // 把 ivec 的元素拷贝给 ivec3vectorstring svec (ivec2) ; // 错误svec 的元素是 string 对象不是 int
列表初始化vector对象
C新标准还提供了另外一种为vector对象的元素赋初值的方法即列表初始化(参见2.2.1节第39页)。此时用花括号括起来的0个或多个初始元素值被赋给vector对象vectorstringarticles(”au”,an,the};上述vector对象包含三个元素第一个是字符串a,第二个是字符串an,最后一个是字符串the.之前已经讲过C语言提供了几种不同的初始化方式(参见2.2.1节第39页)。在大多数情况下这些初始化方式可以相互等价地使用不过也并非一直如此。目前已经介绍过的两种例外情况是其一使用拷贝初始化时(即使用时)(参见3.2.1节第76页),只能提供一个初始值其二如果提供的是一个类内初始值(参见2.6.1节第64页)则只能使用拷贝初始化或使用花括号的形式初始化。第三种特殊的要求是如果提供的是初始元素值的列表则只能把初始值都放在花括号里进行列表初始化而不能放在圆括号里vectorstring vl { naM , an” the”}; // 列表初始化vectorstring v2 (a*1, an”, the); // 错误
创建指定数量的元素
还可以用vector对象容纳的元素数量和所有元素的统一初始值来初始化vector对象:vectorint ivec (10, -1) ; // 10个 int类型的元素每个都被初始化为-1vectorstring svec (10, hi!”) // 10 个 string 类型的元素每个都被初始化为hi!”
值初始化
通常情况下可以只提供Vector对象容纳的元素数量而不用略去初始值。此时库会创建一个值初始化的(value-initialized)元素初值并把它赋给容器中的所有元素。这个初值由vector对象中元素的类型决定。如果vector对象的元素是内置类型比如int,则元素初始值自动设为0。如果元素是某种类类型比如string,则元素由类默认初始化vectorint ivec(10);// 10个元素每个都初始化为0vectorstring svec(10);// 10个元素每个都是空string对象对这种初始化的方式有两个特殊限制:其一,有些类要求必须明确地提供初始值(参见2.2.1节第40页)如果vector对象中元素的类型不支持默认初始化我们就必须提供初始的元素值。对这种类型的对象来说只提供元素的数量而不设定初始值无法完成初始化工作。其二如果只提供了元素的数量而没有设定初始值只能使用直接初始化vectorint vi 10; / / 错误必须使用直接初始化的形式指定向量大小这里的10是用来说明如何初始化vector对象的我们用它的本意是想创建含有10个值 初始化了的元素的vector对象而非把数字10 “拷贝”到 vector中。因此此时不宜使用拷贝初始化7.5.4节 (第 265页)将对这一点做更详细的介绍。
列表初始值还是元素数量
在某些情况下初始化的真实含义依赖于传递初始值时用的是花括号还是圆括号。例如用一个整数来初始化vectorint时整数的含义可能是vector对象的容量也可能是元素的值。类似的用两个整数来初始化vectorint时这两个整数可能一个是vector对象的容量另一个是元素的初值也可能它们是容量为2的vector对象中两个元素的初值。通过使用花括号或圆括号可以区分上述这些含义vectorint vl(10); // vl有 10个元素每个的值都是0vectorint v2(10};// v2有 1个元素该元素的值是10vectorintv3(10, 1);// v3有 10个元素每个的值都是1vectorint v4(10, 1}; // v4有 2个元素值分别是10和 1如果用的是圆括号可以说提供的值是用来构造(construct)vector对象的。例如vl的初始值说明了vector对象的容量v3的两个初始值则分别说明了vector对象的容量和元素的初值。如果用的是花括号可以表述成我们想列表初始化(listinitialize)该vector对象。也就是说初始化过程会尽可能地把花括号内的值当成是元素初始值的列表来处理只有在无法执行列表初始化时才会考虑其他初始化方式。在上例中给v2和v4提供的初始值都能作为元素的值所以它们都会执行列表初始化vector对象v2包含一个元素而vector对象v4包含两个元素。另一方面如果初始化时使用了花括号的形式但是提供的值又不能用来列表初始化就要考虑用这样的值来构造vector对象了。例如要想列表初始化一个含有string对象的vector对象应该提供能赋给string对象的初值。此时不难区分到底是要列表初始化vector对象的元素还是用给定的容量值来构造vector对象尽管在上面的例子中除了第二条语句之外都用了花括号但其实只有v 5 是列表初始化。 要想列表初始化vector对象花括号里的值必须与元素类型相同。显然不能用int初始化string对象所以v7 和 v8 提供的值不能作为元素的初始值。确认无法执行列表初始化后编译器会尝试用默认值初始化vector对象。
3.3.2向vector对象中添加元素
对Vector对象来说直接初始化的方式适用于三种情况初始值已知且数量较少、初始值是另一个vector对象的副本、所有元素的初始值都一样。然而更常见的情况是创建一个vector对象时并不清楚实际所需的元素个数元素的值也经常无法确定。还有些时候即使元素的初值已知但如果这些值总量较大而各不相同那么在创建vector对象的时候执行初始化操作也会显得过于烦琐。举个例子如果想创建一个vector对象令其包含从0到9共10个元素使用列表初始化的方法很容易做到这一点但如果vector对象包含的元素是从0到99或者从0至999呢这时通过列表初始化把所有元素都一一罗列出来就不太合适了。对于此例来说更好的处理方法是先创建一个空vector,然后在运行时再利用vector的成员函数push_back向其中添加元素。push_back负责把一个值当成vector对象的尾元素“压至(push)vector对象的“尾端(back)”。例如在上例中尽管知道vector对象最后会包含100个元素但在一开始还是把它声明成空vector,在每次迭代时才顺序地把下一个整数作为v2的新元素添加给它。同样的如果直到运行时才能知道vector对象中元素的确切个数也应该使用刚刚这种方法创建vector对象并为其赋值。例如有时需要实时读入数据然后将其赋予vector对象和之前的例子一样本例也是先创建一个空vector,之后依次读入未知数量的值并保存至text 中。向vector对象添加元素蕴含的编程假定
由于能高效便捷地向vector对象中添加元素很多编程工作被极大简化了。然而这种简便性也伴随着一些对编写程序更高的要求其中一条就是必须要确保所写的循环正确无误特别是在循环有可能改变vector对象容量的时候。随着对vector的更多使用我们还会逐渐了解到其他一些隐含的要求其中一条是现在就要指出的如果循环体内部包含有向vector对象添加元素的语句则不能使用范围for循环具体原因将在5.4.3节第168页详细解释。范围for语句体内不应改变其所遍历序列的大小 因为vector在扩张的时候指向最后的位置会失效for循环找不到结束判断条件
3.3.3其他vector操作
除了push_back之外vector还提供了几种其他操作大多数都和string的相关操作类似募3.5列出了其中比较重要的一些。访问vector对象中元素的方法和访问string对象中字符的方法差不多也是通过元素在vector对象中的位置。例如可以使用范围for语句处理vector对象中的所 有元素第一个循环把控制变量i定义成引用类型这样就能通过i给V的元素赋值其中i的类型由auto关键字指定。这里用到了一种新的复合赋值运算符参见1.4.1节第10页。如我们所知把左侧运算对象和右侧运算对象相加结果存入左侧运算对象类似的*把左侧运算对象和右侧运算对象相乘结果存入左侧运算对象。最后第二个循环输出所有元素。vector的empty和size两个成员与string的同名成员参见3.2.2节第78页功能完全一致empty检查vector对象是否包含元素然后返回一个布尔值size则返回vector对象中元素的个数返回值的类型是由vector定义的size_type类型。要使用size_type,需首先指定它是由哪种类型定义的、vector对象的类型总是包含着元素的类型参见3.3节第87页 vectorint: : size_type // 正确vector: :size_type // 错误各个相等性运算符和关系运算符也与string的相应运算符参见3.2.2节第79页功能一致。两个vector对象相等当且仅当它们所含的元素个数相同而且对应位置的元素值也相同。关系运算符依照字典顺序进行比较如果两个vector对象的容量不同但是在相同位置上的元素值都一样则元素较少的vector对象小于元素较多的vector对象;若元素的值有区别则vector对象的大小关系由第一对相异的元素值的大小关系决定。只有当元素的值可比较时vector对象才能被比较。一些类如string等确实定义了自己的相等性运算符和关系运算符另外一些如Sales_item类支持的运算已经全都罗列在1.5.1节第17页中了显然并不支持相等性判味和关系运算等操作。因此不能比较两个vectorSales_item对象。
计算vector内对象的索引
使用下标运算符参见323节第84页能获取到指定的元素。和string-样vector对象的下标也是从0开始计起下标的类型是相应的size_type类型。只要vector对象不是一个常量就能向下标运算符返回的元素赋值。此外如3.2.3节第85页所述的那样也能通过计算得到vector内对象的索引然后直接获取索引位置上的元素。举个例子假设有一组成绩的集合其中成绩的取值是从0到100。以10分为一个分数段要求统计各个分数段各有多少个成绩。显然从0到100总共有101种可能的成绩取值这些成绩分布在11个分数段上每10个分数构成一个分数段这样的分数段有10个额外还有一个分数段表示满分100分。这样第一个分数段将统计成绩在0到9之间的数量第二个分数段将统计成绩在10到19之间的数量以此类推。最后一个分数段统计满分100分的数量。按照上面的描述如果输入的成绩如下42 65 95 100 39 67 95 76 88 76 83 92 76 93则输出的结果应该是:0 0 0 1 1 0 2 3 2 4 1结果显示成绩在30分以下的没有、30分至39分有1个、40分至49分有1个、50分至59分没有、60分至69分有2 个、70分至79分有3 个、80分至89分有2 个、90分至99分有4 个还有1个是满分。在具体实现时使用一个含有11个元素的vector对象每个元素分别用于统计各个分数段上出现的成绩个数。对于某个成绩来说将其除以10就能得到对应的分数段索引。注意两个整数相除结果还是整数余数部分被自动忽略掉了。例如42/104、65/106、100/1010等。一旦计算得到了分数段索引取该分数段的计数值并加1:就能用它作为vector对象的下标进而获取该分数段的计数值并加1:在上面的程序中首先定义了一个Vector对象存放各个分数段上成绩的数量。此例中由于初始状态下每个元素的值都相同所以我们为vector对象申请了11个元素并把所有元素的初始值都设为0。while语句的条件部分负责读入成绩在循环体内部首先检查读入的成绩是否合法即是否小于等于100分如果合法将成绩对应的分数段的计数值加1。执行计数值累加的那条语句很好地体现了C程序代码的简洁性。
不能用下标形式添加元素
刚接触C语言的程序员也许会认为可以通过vector对象的下标形式来添加元素, 事实并非如此。下面的代码试图为vector对象ivec添加10个元素vectorint ivec; // 空 vector 对象