目录
数组是一种数据格式,能够存储多个同类型的值。要创建数组,可以使用声明语句。数组声明应指出以下三点:
(1) 存储在每个元素中的值的类型
(2) 数组名
(3) 数组中的元素数
在C++中,可以通过修改简单变量的声明,添加中括号(其中包含元素数目)来完成数组声明。例如,下面的声明创建一个名为months的数组,该数组有12个元素,每个元素都可以存储一个short类型的值:
short months[12];
事实上可以将数组中的每一个元素看成一个简单变量。声明数组的通用格式如下:
typeName arrayName[arraySize];
表达式arraySize指定元素数目,他必须是整型常量(如10)或const值,也可以是常量表达式,即其中所有的值在编译过程中都是已知的。具体地说arraySize不能是变量,变量的值是在程序运行时设置的。
数组之所以被称为复合类型,是因为他是使用其他类型来创建的。不能仅仅将某种东西声明为数组,他必须是特定类型的数组。
数组的很多用途都是基于这样一个事实:可以单独访问数组元素。方法是通过下标或索引来对数组元素进行编号。C++数组从0开始编号,C++使用带索引的方括号表示法来指定数组元素。
注意:最后一个元素的索引比数组长度小1。
有效下标值的重要性:编译器不会检查使用的下标是否有效。例如,如果将一个值赋给不存在的元素months[101],编译器并不会指出错误。但是程序运行后,这种赋值可能会引发问题。
#include<iostream>
int main() {
using namespace std;
int yams[3];
yams[0] = 7;
yams[1] = 8;
yams[2] = 6;
int yamcosts[3] = { 20,30,5 };
cout << "Total yams = ";
cout << yams[0] + yams[1] + yams[2] << endl;
cout << "The package with " << yams[1] << " yams costs ";
cout << yamcosts[1] << " cents per yam.\n";
int total = yams[0] * yamcosts[0] + yams[1] * yamcosts[1];
total = total + yams[2] * yamcosts[2];
cout << "The total yam expense is " << total << " cents.\n";
cout << "\nSize of yams array = " << sizeof yams;
cout << " bytes.\n";
cout << "Size of one element = " << sizeof yams[0];
cout << " bytes.\n";
return 0;
}
3.1.1 程序说明
程序给yam的元素赋值时,绕了一个大弯。C++允许在声明语句中初始化数组元素。程序中使用这种捷径来给yamcosts数组赋值:
int yamcosts[3] = { 20,30,5 };
只需要提供一个用逗号分隔的值列表(初始化列表),用花括号将他们括起来即可。列表中的空格是可选的。
3.1.2 数组的初始化规则
只有在定义数组时才能使用初始化,以后就不能使用了,也不能将一个数组赋给另一个数组:
int cards[4] = {3,6,8,10}; //创建一个初始化数组
int hand[4]; //创建了一个未初始化的数组
hand[4] = {5,6,7,8}; //定义数组的时候才能初始化,以后就不能再进行初始化了
hand = cards; //不能将一个数组赋给另外一个数组
然而,可以使用数组中的下标给数组中的元素赋值,初始化数组时,提供的值可以少于数组的元素数目。例如,下面的语句只初始化hotelTips的前两个元素:
float hotelTips[5] = {5.0, 2.5};
如果只对数组的一部分元素进行初始化,则编译器将把其他元素设置为0。因此将数组中所有元素都初始化为0非常简单——只要显式的将第一个元素初始化为0,然后让编译器将其他元素都初始化为0即可:
long totals[500] = {0};
如果初始化为{1}而不是{0},则第一个元素被设置为1,其他元素都被设置为0。如果初始化数组时方括号内([])为空,C++编译器将计算元素个数。例如,对于下面的声明:
short things[] = {1, ,5, 3, 8};
编译器将使things数组包含4个元素。
3.1.3 C++11数组初始化方法
前面也说过,C++11将使用大括号的的初始化(列表初始化)作为一种通用的初始化方法,可用于所有类型。数组以前就可以使用列表初始化,但C++11的列表初始化新增了一些功能。
首先,初始化数组时,可以省略等号(=):
double earnings[4]{1.2e4, 1.6e4, 1.1e4, 1.7e4};
其次,可不在大括号内包含任何东西,这把所有元素都设置为0:
unsigned int counts[10] = {};
float balances[100]{};
第三,列表初始化禁止缩窄转换:
long plifs[] = {25, 92, 3.0}; //not allowed 浮点数转换为整型是缩窄操作,即使浮点数的小数点后面是0
char slifs[4]{'h', 'i', 1122011, '\0'}; //not allowed 1122011超出了char变量的取值范围(这里假设char变量长度范围为8位)
char tlifs[4]{'h', 'i', 112, '\0'}; //allowed 可以通过编译,虽然112是一个int值,但是它在char变量的取值范围内