C++primer 第 3 章 字符串、向量和数组 3 . 3 标准库类型vector

  • 标准库类型vector表示对象的集合,其中所有对象的类型都相同。集合中的每个对象都有一个与之对应的索引,索引用于访问对象。因为vector"容纳着”其他对象,所以它也常被称作容器(container).第 II部将对容器进行更为详细的介绍。 要想使用vector,必须包含适当的头文件。在后续的例子中,都将假定做了如下using声明:
  • #include <vector>
  • using std::vector;
  • C++语言既有类模板(classtemplate),也有函数模板,其中vector是一个类模板。只有对C++有了相当深入的理解才能写出模板,事实上,我们直到第16章才会学习如何自定义模板。幸运的是,即使还不会创建模板,我们也可以先试着用用它。
  • 模板本身不是类或函数,相反可以将模板看作为编译器生成类或函数编写的一份说明。编译器根据模板创建类或函数的过程称为实例化(instantiation),当使用模板时,需要指出编译器应把类或函数实例化成何种类型。
  • 对于类模板来说,我们通过提供一些额外信息来指定模板到底实例化成什么样的类,需要提供哪些信息由模板决定。提供信息的方式总是这样:即在模板名字后面跟一对尖括号,在括号内放上信息。

  • vector能容纳绝大多数类型的对象作为其元素,但是因为引用不是对象(参见2.3.1节,第45页),所以不存在包含引用的vectoro除此之外,其他大多数(非引用)内置类型和类类型都可以构成vector对象,甚至组成vector的元素也可以是vector。需要指出的是,在早期版本的C++标准中如果vector的元素还是vector(或者其他模板类型),则其定义的形式与现在的C++11新标准略有不同。过去,必须在外层vector对象的右尖括号和其元素类型之间添加一个空格,如应该写成vector<vector<int> >而非vector<vector<int>>现在大可不必

3.3.1定义和初始化vector对象

  • 和任何一种类类型一样,vector模板控制着定义和初始化向量的方法。表3.4列出了定义vector对象的常用方法。

  • 可以默认初始化vector对象(参见2.2.1节,第 40页),从而创建一个指定类型的空 vector:
  • vector<string> svec; //默认初始化,svec不含任何元素  最好使用空的花括号进行默认初始化 vector<string> svec{};
  • 看起来空vector好像没什么用,但是很快我们就会知道程序在运行时可以很高效地往 vector对象中添加元素。事实上,最常见的方式就是先定义一个空vector,然后当运行时获取到元素的值后再逐一添加。当然也可以在定义vector对象时指定元素的初始值。例如,允许把一个vector对 象的元素拷贝给另外一个vector对象。此时,新 vector对象的元素就是原vector 对象对应元素的副本。注意两个vector对象的类型必须相同
  • vector<int> ivec; / / 初始状态为空
  • / / 在此处给ivec添加一些值
  • vector<int> ivec2 (ivec) ; // 把 ivec 的元素拷贝给 ivec2
  • vector<int> ivec3 = ivec; // 把 ivec 的元素拷贝给 ivec3
  • vector<string> svec (ivec2) ; // 错误:svec 的元素是 string 对象,不是 int

列表初始化vector对象

  • C++新标准还提供了另外一种为vector对象的元素赋初值的方法,即列表初始化(参见2.2.1节,第39页)。此时,用花括号括起来的0个或多个初始元素值被赋给vector对象
  • vector<string>articles=(”au”,"an","the'};
  • 上述vector对象包含三个元素:第一个是字符串"a",第二个是字符串"an",最后一个是字符串"the".
  • 之前已经讲过,C++语言提供了几种不同的初始化方式(参见2.2.1节,第39页)。在大多数情况下这些初始化方式可以相互等价地使用,不过也并非一直如此。目前已经介绍过的两种例外情况是:其一,使用拷贝初始化时(即使用=时)(参见3.2.1节,第76页),只能提供一个初始值;其二,如果提供的是一个类内初始值(参见2.6.1节,第64页),则只能使用拷贝初始化或使用花括号的形式初始化。第三种特殊的要求是,如果提供的是初始元素值的列表,则只能把初始值都放在花括号里进行列表初始化,而不能放在圆括号里
  • vector<string> vl { naM , "an”, "the”}; // 列表初始化
  • vector<string> v2 ("a*1, "an”, "the"); // 错误

创建指定数量的元素

  • 还可以用vector对象容纳的元素数量和所有元素的统一初始值来初始化vector对象:
  • vector<int> ivec (10, -1) ; // 10个 int类型的元素,每个都被初始化为-1
  • vector<string> svec (10, "hi!”); // 10 个 string 类型的元素,每个都被初始化为"hi!”

值初始化

  • 通常情况下,可以只提供Vector对象容纳的元素数量而不用略去初始值。此时库会创建一个值初始化的(value-initialized)元素初值,并把它赋给容器中的所有元素。这个初值由vector对象中元素的类型决定。如果vector对象的元素是内置类型,比如int,则元素初始值自动设为0。如果元素是某种类类型,比如string,则元素由类默认初始化
  • vector<int> ivec(10);// 10个元素,每个都初始化为0
  • vector<string> svec(10);// 10个元素,每个都是空string对象
  • 对这种初始化的方式有两个特殊限制:
  • 其一,有些类要求必须明确地提供初始值(参见2.2.1节,第40页),如果vector对象中元素的类型不支持默认初始化,我们就必须提供初始的元素值。对这种类型的对象来说,只提供元素的数量而不设定初始值无法完成初始化工作。
  • 其二,如果只提供了元素的数量而没有设定初始值,只能使用直接初始化
  • vector<int> vi = 10; / / 错误:必须使用直接初始化的形式指定向量大小
  • 这里的10是用来说明如何初始化vector对象的,我们用它的本意是想创建含有10个值 初始化了的元素的vector对象,而非把数字10 “拷贝”到 vector中。因此,此时不宜使用拷贝初始化,7.5.4节 (第 265页)将对这一点做更详细的介绍。

列表初始值还是元素数量?

  • 在某些情况下,初始化的真实含义依赖于传递初始值时用的是花括号还是圆括号。例如,用一个整数来初始化vector<int>时,整数的含义可能是vector对象的容量也可能是元素的值。类似的,用两个整数来初始化vector<int>时,这两个整数可能一个是vector对象的容量,另一个是元素的初值,也可能它们是容量为2的vector对象中两个元素的初值。通过使用花括号或圆括号可以区分上述这些含义
  • vector<int> vl(10); // vl有 10个元素,每个的值都是0
  • vector<int> v2(10};// v2有 1个元素,该元素的值是10
  • vector<int>v3(10, 1);// v3有 10个元素,每个的值都是1
  • vector<int> 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页 ):
  • vector<int>: : size_type // 正确
  • vector: :size_type // 错误
  • 各个相等性运算符和关系运算符也与string的相应运算符(参见3.2.2节,第79页)功能一致。两个vector对象相等当且仅当它们所含的元素个数相同,而且对应位置的元素值也相同。关系运算符依照字典顺序进行比较:如果两个vector对象的容量不同,但是在相同位置上的元素值都一样,则元素较少的vector对象小于元素较多的vector对象;若元素的值有区别,则vector对象的大小关系由第一对相异的元素值的大小关系决定。只有当元素的值可比较时,vector对象才能被比较。一些类,如string等,确实定义了自己的相等性运算符和关系运算符;另外一些,如Sales_item类支持的运算已经全都罗列在1.5.1节(第17页)中了,显然并不支持相等性判味和关系运算等操作。因此,不能比较两个vector<Sales_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/10=4、65/10=6、100/10=10等。一旦计算得到了分数段索引,取该分数段的计数值并加1:就能用它作为vector对象的下标,进而获取该分数段的计数值并加1:

  • 在上面的程序中,首先定义了一个Vector对象存放各个分数段上成绩的数量。此例中,由于初始状态下每个元素的值都相同,所以我们为vector对象申请了11个元素,并把所有元素的初始值都设为0。while语句的条件部分负责读入成绩,在循环体内部首先检查读入的成绩是否合法(即是否小于等于100分),如果合法,将成绩对应的分数段的计数值加1。
  • 执行计数值累加的那条语句很好地体现了C++程序代码的简洁性。

不能用下标形式添加元素

  • 刚接触C++语言的程序员也许会认为可以通过vector对象的下标形式来添加元素, 事实并非如此。下面的代码试图为vector对象ivec添加10个元素:
  • vector<int> ivec; // 空 vector 对象

 

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

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

相关文章

SpringBoot AOP切面实现

文章目录一、AOP简介二、AOP体系与概念三、AOP实例1、创建SpringBoot工程2、添加依赖3、AOP相关注解3.1、Aspect3.2、Pointcut3.2.1、execution()3.2.2、annotation()3.3、Around3.4、Before3.5、After3.6、AfterReturning3.7、AfterThrowing一、AOP简介 AOP&#xff08;Aspec…

英语口语-文章朗读Week8 Friday

文章 It is a phenomenon that people are losing trust in each other in today’s society. Some people become selfish,and for interest, they are likely to betray their colleagues,friends, and even their relatives. They tend to cater to those who can benefit …

C++primer 第 3 章 字符串、向量和数组 3 . 4 迭代器介绍

3.4迭代器介绍 我们已经知道可以使用下标运算符来访问string对象的字符或vector对象的元素&#xff0c;还有另外一种更通用的机制也可以实现同样的目的&#xff0c;这就是迭代器&#xff08;iterator&#xff09;。在第II部分中将要介绍&#xff0c;除了vector之外&#xff0c…

英语口语-文章朗读Week9 TuesDay

朗读文章 People living in ancient times had no alternative but to do housework manually. They fire the wood when they cook,they hand wash clothes with hands; they sweep the floor with brooms. Now, modern inventions come as a great relief to people. We co…

C++primer 第 3 章 字符串、向量和数组 3 . 5 数组

3.5数组 数组是一种类似于标准库类型vector&#xff08;参见3.3节&#xff0c;第86页&#xff09;的数据结构&#xff0c;但是在性能和灵活性的权衡上又与vector有所不同。与vector相似的地方是&#xff0c;数组也是存放类型相同的对象的容器&#xff0c;这些对象本身没有名字…

C++primer 第 4 章 表达式 4.1基础 4 . 2 算术运算符 4 .3 逻辑和关系运算符 4 . 4 赋值运算符 4 .5 递增和递减运算符 4.6成员访问运算符

表达式由一个或多个运算对象(operand)组成&#xff0c;对表达式求值将得到一个结果(result)字面值和变量是最简单的表达式(expression),其结果就是字面值和变量的值。把一个运算符(operator)和一个或多个运算对象组合起来可以生成较复杂的表达式 4.1基础 有几个基础概念对表达…

codeforces 266B-C语言解题报告

266B题目网址 题目解析 输入n,t,排队情况s,输出第t次循环后,排队情况 举例: 输入: 5 1 BGGBG 输出: GBGGB 2.输入的n代表排队的人数,t代表整个循环t次之后再输出结果 3.注意点: 使用while()大循环去控制t次的循环,使用for()内层循环去遍历整个字符串 如果if(s[j]‘B’&…

英语口语-文章朗读Week9 Wednesday

英语文章 Birds of the same species flock together&#xff0c; People tend to look for someone like themselves to be friends. But having the same interests is not the only standard when we are seeking friends. In most cases, especially for adults, people l…

C++primer 第 4 章 表达式 4.7条件运算符 4.8位运算符 4.9 sizeof运算符 4.10逗号运算符 4.11类型转换 4 . 1 2 运算符优先级表

4.7条件运算符 条件运算符(?&#xff1a;)允许我们把简单的if else逻辑嵌入到单个表达式当中&#xff0c;条件运算符按照如下形式使用&#xff1a;cond ? expr1 : expr2;其中cond是判断条件的表达式&#xff0c;而expr1和expr2是两个类型相同或可能转换为某个公共类型的表达…

Git 之 git tag标签使用

目录一、简介二、本地tag操作1、创建tag标签&#xff08;1&#xff09;创建轻量标签&#xff08;2&#xff09;创建附注标签2、查看tag标签&#xff08;1&#xff09;查看标签列表&#xff08;2&#xff09;查看标签提交信息&#xff08;3&#xff09;在提交历史中查看标签3、删…

C++primer 第 5 章语句 5.2语句作用域 5.3条件语句 5 . 4 迭代语句 5.5跳转语句 5.6 try语句块和异常处理

5 . 1 简单语句 C语言中的大多数语句都以分号结束&#xff0c;一个表达式&#xff0c;比如ival 5 , 末尾加上分号就变成了表达式语句(expression statement)。表达式语句的作用是执行表达式并丢弃掉求值结果&#xff1a;ival 5&#xff1b; // 一条没什么实际用处的表达式语…

英语口语-文章朗读Week9Thursday

英语文章 Everyone has his or her own dreams. Some people wants to be millionaires so they can give many generous donations later; some people want to be scientists so they can bring many conveniences to the world; some people only want to be bus-drivers s…

操作系统 内存管理相关知识

cpu执行程序的基本过程 译码器 输入为n管脚&#xff0c;输出为2^n根管脚&#xff0c;编号为从0到2^(n-1)&#xff0c;用少的输入端控制更多的输出端最常用的是三八译码器AD(Address bus)地址总线: 选中一行数据每一行 8bit 组成8吧B cpu输入端32根线&#xff0c;输出端就可以控…

Chrome浏览器必装插件!尤其程序猿!

Chrome 浏览器有一个好处&#xff0c;就是插件极其丰富&#xff0c;只有你想不到的&#xff0c;没有你找不到的&#xff0c;这恐怕是 Chrome 浏览器被众多爱好者钟爱的原因吧。 言归正传&#xff0c;今天来给大家推荐 10 款我自己珍藏的 Chrome 浏览器插件。 1、crxMouse Ch…

英语口语-文章朗读Week10 Monday

英语文章 Here are some valuable suggestions which may assist you in landing good job First, make your resume clear and associate it with the position you are applying for. Try to add details like your temporary jobs at college or your former jobs Second, …

英语口语-文章朗读Week10 Wednesday

英语文章 Everyone needs sleep for survival, but how much? It is said that eight hours of sleep is fundamental to a healthy person. But today, many people are sacrificing their sleep time。 Modern people have so many alternatives: cell phones, PCs, TVs, g…

嵌入式Linux多任务编程 进程 管道 命名管道

进程 进程是一个可并发执行的具有独立功能的程序关于某个数据集合的一次执行过程&#xff0c;也是操作系统执行资源分配和保护的基本单位。程序的一次执行就是一个进程一个程序可以派生多个进程多个不同程序运行的时候&#xff0c;也会有多个相对应的进程与其相互对应进程是动…

英语口语-文章朗读Week10 Thursday

英语文章 There are many customs and traditions in Chinese civilization. Here, we will talk about the development of the way people greet each other: In ancient times, people had to kneel to those who were superior to them. This custom remained until the …

Linux进程之间通信 信号

2) SIGINT 程序终止(interrupt)信号, 在用户键入INTR字符(通常是Ctrl-C)时发出&#xff0c;用于通知前台进程组终止进程。 3) SIGQUIT 和SIGINT类似, 但由QUIT字符(通常是Ctrl-\)来控制. 进程在因收到SIGQUIT退出时会产生core文件, 在这个意义上类似于一个程序错误信号。 15)…