和任何一种类类型一样,vector模板控制着定义和初始化向量的方法。表3.4列出了定义vector对象的常用方法。
表3.4:初始化vector对象的方法
表3.4:初始化vector对象的方法 | |
vector<T> v1 | v1是一个空vector,它潜在的元素是T类型的,执行默认初始化 |
vector<T> v2(v1) | v2中包含有v1所有元素的副本 |
vector<T> v2 = v1 | 等价于v2(v1),v2中包含有v1所有元素的副本 |
vector<T> v3(n, val) | v3包含了n个重复的元素,每个元素的值都是val |
vector<T> v4(n) | v4包含了n个重复地执行了值初始化的对象 |
vector<T> v5{a,b,c...} | v5包含了初始值个数的元素,每个元素被赋予相应的初始值 |
vector<T> v5={a,b,c...} | 等价于v5{a,b,c...} |
可以默认初始化vector对象(参见2.2.1节,第44页),从而创建一个指定类型的空vector:
- vector<string> svec; //默认初始化,svec不含任何元素
看起来空vector好像没什么用,但是很快我们就会知道程序在运行时可以很高效地往vector对象中添加元素。事实上,最常见的方式就是先定义一个空vector,然后当运行时获取到元素的值后再逐一添加。
当然也可以在定义vector对象时指定元素的初始值。例如,允许把一个vector对象的元素拷贝给另外一个vector对象。此时,新vector对象的元素就是原vector对象对应元素的副本。注意两个vector对象的类型必须相同:
- vector<int> ivec; // 初始状态为空
- // 在此处给ivec添加一些值
- vector<int> ivec2(ivec); // 把ivec的元素拷贝给 ivec2
- vector<int> ivecivec3 = ivec; // 把ivec的元素拷贝给 ivec3
- vector<string> svec(ivec2); // 错误:svec的元素是string对象,不是int
列表初始化vector对象
C++11新标准还提供了另外一种为vector对象的元素赋初值的方法,即列表初始化(参见2.2.1节,第43页)。此时,用花括号括起来的0个或多个初始元素值被赋给vector对象:
- vector<string> articles = {"a", "an", "the"};
上述vector对象包含三个元素:第一个是字符串"a",第二个是字符串"an",最后一个是字符串"the"。
之前已经讲过,C++语言提供了几种不同的初始化方式(参见2.2.1节,第43页)。在大多数情况下这些初始化方式可以相互等价地使用,不过也并非一直如此。目前已经介绍过的两种例外情况是:其一,使用拷贝初始化时(即使用=时)(参见3.2.1节,第84页),只能提供一个初始值;其二,如果提供的是一个类内初始值(参见2.6.1节,第73页),则只能使用拷贝初始化或使用花括号的形式初始化。第三种特殊的要求是,如果提供的是初始元素值的列表,则只能把初始值都放在花括号里进行列表初始化,而不能放在圆括号里:
- vector<string> v1{"a", "an", "the"}; // 列表初始化
- vector<string> v2("a", "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节,第44页),如果vector对象中元素的类型不支持默认初始化,我们就必须提供初始的元素值。对这种类型的对象来说,只提供元素的数量而不设定初始值无法完成初始化工作。
其二,如果只提供了元素的数量而没有设定初始值,只能使用直接初始化:
- vector<int> vi = 10; // 错误:必须使用直接初始化的形式指定向量大小