您的位置:首页 > 其它

定义和初始化vector对象

2015-08-30 10:00 141 查看
和任何一种类类型一样,vector模板控制着定义和初始化向量的方法。下面列出了定义vector对象的常用方法。

**默认初始化**vector对象从而创建一个指定类型的空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> ivec3 = ivec;    // 把ivec的元素拷贝给 ivec3

vector<string> svec(ivec2); // 错误:svec的元素是string对象,不是int


列表初始化vector对象

C++11新标准还提供了另外一种为vector对象的元素赋初值的方法,即列表初始化。此时,用花括号括起来的0个或多个初始元素值被赋给vector对象:

vector<string> articles = {"a", "an", "the"};


上述vector对象包含三个元素:第一个是字符串”a”,第二个是字符串”an”,最后一个是字符串”the”。

之前已经讲过,C++语言提供了几种不同的初始化方式。在大多数情况下这些初始化方式可以相互等价地使用,不过也并非一直如此。目前已经介绍过的两种例外情况是:其一,使用拷贝初始化时(即使用=时)只能提供一个初始值;其二,如果提供的是一个类内初始值,则只能使用拷贝初始化或使用花括号的形式初始化。第三种特殊的要求是,如果提供的是初始元素值的列表,则只能把初始值都放在花括号里进行列表初始化,而不能放在圆括号里:

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; // 错误:必须使用直接初始化的形式指定向量大小


这里的10是用来说明如何初始化vector对象的,我们用它的本意是想创建含有10个值初始化了的元素的vector对象,而非把数字10“拷贝”到vector中。因此,此时不宜使用拷贝初始化,将对这一点做更详细的介绍。

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

在某些情况下,初始化的真实含义依赖于传递初始值时用的是花括号还是圆括号。例如,用一个整数来初始化vector时,整数的含义可能是vector对象的容量也可能是元素的值。类似的,用两个整数来初始化vector时,这两个整数可能一个是vector对象的容量,另一个是元素的初值,也可能它们是容量为2的vector对象中两个元素的初值。通过使用花括号或圆括号可以区分上述这些含义:

vector<int> v1(10);     // v1有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对象的。例如,v1的初始值说明了vector对象的容量;v3的两个初始值则分别说明了vector对象的容量和元素的初值。

如果用的是花括号,可以表述成我们想列表初始化(list initialize)该vector对象。也就是说,初始化过程会尽可能地把花括号内的值当成是元素初始值的列表来处理,只有在无法执行列表初始化时才会考虑其他初始化方式。在上例中,给v2和v4提供的初始值都能作为元素的值,所以它们都会执行列表初始化,vector对象v2包含一个元素而vector对象v4包含两个元素。

另一方面,如果初始化时使用了花括号的形式但是提供的值又不能用来列表初始化,就要考虑用这样的值来构造vector对象了。例如,要想列表初始化一个含有string对象的vector对象,应该提供能赋给string对象的初值。此时不难区分到底是要列表初始化vector对象的元素还是用给定的容量值来构造vector对象:

vector<string> v5{"hi"}; // 列表初始化:v5有一个元素

vector<string> v6("hi"); // 错误:不能使用字符串字面值构建vector对象

vector<string> v7{10};          // v7有10个默认初始化的元素

vector<string> v8{10, "hi"};    // v8有10个值为"hi"的元素


尽管在上面的例子中除了第二条语句之外都用了花括号,但其实只有v5是列表初始化。要想列表初始化vector对象,花括号里的值必须与元素类型相同。显然不能用int初始化string对象,所以v7和v8提供的值不能作为元素的初始值。确认无法执行列表初始化后,编译器会尝试用默认值初始化vector对象。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: