打赏

把《c++ primer》读薄(4-1 c和c++数组)

督促读书,总结精华,提炼笔记,抛砖引玉,有不合适的地方,欢迎留言指正。

c和c++的数组和指针都属于低级的复合数据类型,比如c++的数组,类似vector容器,指针类似迭代器。低级的数据类型优势是速度快。但是容易出错,不好调试。现代c++程序,应该避免使用。

内置数据类型—数组,不方便存储变长数据,定义之后长度固定(静态数组),数组类似容器,比如,无法事先知道一个给定数组的长度,也没有类似size函数来求数组长度,也没有push_back()函数来添加元素,如果要更改数组长度,程序员只能重新建立数组,把旧的复制到新的。

记住现代c++程序,应该尽量是vector容器来替代数组,来存取同一类型的数据元素,只有当要求速度的前提下,如果使用容器不达标,那么再使用数组。

问题1、array由一系列的类型相同的元素构成,数组声明包括数组元素个数和类型,数组最好定义之后就初始化。

数组的声明

    //[]方括号表示声明的是数组,里面的数字表明了数组包含的元素数目
    int states[50];//声明50个整数的数组
    double code[365];//声明365个浮点数的数组
    char chr[20];//声明20个字符的数组

访问数组内容可以用下标表示单个元素,下标数字也叫索引(index),从0开始计算。

数组的初始化

    //用花括号括起来,元素用逗号隔开
    int pow[8] = {1, 2, 3, 4, 5, 6, 7, 8};//只有标准c,c++支持此初始化语法
    //pow[0]首元素赋值为1,以此类推

注意:数组声明的时候最好是初始化,虽然不会报错,但是和普通变量一样,使用没有初始化的数组,里面元素的值不定,出现垃圾值。

数组初始化列表和大小不一致的情况

初始化列表里的元素个数和数组大小不一致,当初始化元素数目少于数组大小,多余的元素自动初始化为0。如果没有初始化数组,那么和普通变量一样,存取的内存原来就有的垃圾值。初始化了,只不过部分初始化,那么编译器会自动初始化余下的元素为0。初始化了,但是初始化列表元素大于数组大小,那么编译出报错。

    //初始化没有完全,编译器自动后面给初始化为0
    double d[4] = {1};//不过vs2010里编译运行,全都是0

    //error C2078: 初始值设定项太多
    double d[4] = {1, 2, 3, 4, 5, 6}; 

数组初始化小技巧

省略填写数组大小,让编译器自动的去判断数组的实际大小和初始化列表里的项目

    //空的方括号告诉编译器去初始化列表里判断数组实际大小
    char chr[] = {'a', 'b', ' ', '4'};
    //这里需要一个技巧,人工判断数组大小容易出错,使用sizeof运算符计算
    for (int i = 0; i < sizeof(chr) / sizeof(chr[0]); i++)
    {
        cout << "i + 1 = " << chr[i] << endl;
    }

注意技巧:sizeof(数组名)求的是数组总共多大(字节单位),sizeof(数组【0】)求的是数组中一个元素的大小(字节单位),两者除法,就是数组的实际大小(元素个数)。

数组使用小技巧(常量大小)

    const int NUM = 4;//常量代表数组大小,推荐的技巧
    int days[NUM] = {1, 2, 3, 4};//良好的编程风格,如果以后想修改数组大小,只需要修改开头的常量即可

    for (int index = 0; index < NUM; index++)
    {
        cout << days[index] << endl;
    }

对数组指定元素初始化

属于C99的新特性,可以对数组指定的元素直接初始化,如果对数组最后一个元素初始化,那么传统语法必须顺次初始化到最后一个元素前,才能对最后一个元素初始化。

    //对最后一个元素初始化为1
    float f[3] = {0, 0, 1};

而c99规定可以直接去初始化

    //使用方括号【】,直接对数组某个元素初始化
    int i[3] = {i[2] = 100};//ok!需要加  数组名【】
    //int in[5] = {[4] = 10};//error C2059: 语法错误:“[”

对于一般的数组初始化,部分初始化之后,余下的自动初始化为0(这里vs2010编译器不是这样的,全部都是0 了),如果在指定的初始化元素后还有别的初始化值,那么这些数值自动对数组后续元素进行初始化,并且指定初始化元素的数值在第几个元素处,那么也会同样初始化这个元素。

比如in【0】=11在第5(0-5)个元素位置,那么同时打印第5个元素=11。说明初始化指定元素的同时,也用这个数值初始化本位置的元素了。

如果多次对数组某个元素初始化,则最后一次为准。

    int in[7] = {1, 2, in[2] = 10, 3, 4, in[0] = 11};   //首元素i0=11,i5=11
    int i;

    for (i = 0; i < 7; i++)
    {
        cout << "i" << i << " = " << in[i] << endl;
    }

只读数组

如果只需要对数组读取,而不进行修改,那么推荐关键字const

const int days[NUM] = {1, 2, 3, 4};//数组中每个元素都当作常量处理,和普通变量一样,const数组也需要声明的时候初始化

数组的赋值

使用数组下标(索引)为数组元素赋值,c和c++不支持整体赋值的初始化,或者就是单纯的赋值,也不支持用花括号括起来的列表进行赋值(初始化除外)

    double d[SIZE] = {1, 2, 3};//ok这是可以的,列表进行初始化
    int i[SIZE];

    for (int count = 0; count < SIZE; count++)
    {
        i[count] = 2 * count;//ok,使用循环和数组下标给数组元素赋偶数值
    }

    //i = d;//error,c不支持数组整体赋值
    //i[SIZE] = d[SIZE];//error

    //d[SIZE] = {11, 22, 33};//不起作用,这种形式只有初始化可以使用,赋值必须使用索引

数组的边界问题

千万不要越界(类似数据类型的越界),数组索引不能越界,因为编译器不会检测这种错误。编译器不会检测索引的合法性,如果出现非法的索引,那么结果未知,有时候会中断,但是也有时候可以执行,但是结果很奇怪(编译器不同而不同),编译器不检测数组边界,是出于信任程序员,可以节省时间和效率。 

注意:记住,数组计数原则上都从0开始,并且数组大小用符号常量代替。减少错误的发生。

再看数组大小的指定

前面可以用整型常量(unsigned)或者字符常量来指定大小,C99之前就是这两种方法。

    const int const_m = 10;//在c语言(不同于c++)里,const值不被看作是常量,在c++里,这样的定义的变量,也就是const常量,可以去做数组的维数。
    int n = 100;//定义了变量n

    double d1[10];// ok
    double d2[5 * 2 + 1];//ok
    double d3[];//error,没有初始化,也没有大小指定
    double d4[sizeof(int)];//ok,sizeof表达式在c里被认为返回一个整数常量
    double d5[-10];//error C2118: 负下标
    double d6[0];//error C2466: 不能分配常量大小为 0 的数组
    double d7[3.14];// error C2058: 常量表达式不是整型
    double d8[(int)3.14];//ok

    double d9[const_m];// error C2057: 应输入常量表达式,“d9”未知的大小,不能分配常量大小为 0 的数组
    double d10[n];//error C2057: 应输入常量表达式,“d9”未知的大小,不能分配常量大小为 0 的数组

c99之后,后两种方式可以了,并且创建了一种新数组VLA(variable length array)变长数组,VLA。目的是为了让c更适合做数值计算。

问题2、下列语法是不合法的

int ia[get_size()];

非法,get_size()是函数调用,不是常量表达式,不能用于定义数组的维数(维长度)。记住,必须是程序编译期间就得确定数组的大小,运行期间的不行。

问题3、注意下列数组的初始值

//ia 为在函数体外定义的内置数组,也就是全局变量,各元素初始化为0
int ia[10];
//sa为元素类型为string 的数组,自动调用string 类的默认构造函数将各元素初始化为空字符串
string sa[10];

int main(void)
{
    //sa2 为元素类型为string 的数组,自动调用string 类的默认构造函数将各元素初始化为空字符串
    string sa2[10];

    //ia2 为在函数体内定义的内置数组,各元素未初始化,其值不确定
    int ia2[10];

    system("pause");
    return 0;
}

问题4、不要把数组和vector容器混淆

    //vector 对象不能用这种方式进行初始化。
    //vector<int> ivec = {0, 1, 1, 2, 3, 5, 8};
    int ia3[] = ivec;//错误。不能用vector 对象来初始化数组。

定义数组时可使用初始化列表来初始化数组的部分或全部元素。如果是初始化全部元素,可以省略定义数组时方括号中给出的数组维数值。

如果指定了数组维数,则初始化列表提供的元素个数不能超过维数值。如果数组维数大于列出的元素初值个数,则只初始化前面的数组元素,剩下的其他元素,若是内置类型则初始化为0,若是类类型则调用该类的默认构造函数进行初始化。

字符数组既可以用一组由花括号括起来、逗号隔开的字符字面值进行初始化,也可以用一个字符串字面值进行初始化。

问题5、列出使用数组而不是vector 的缺点

与vector 类型相比,数组具有如下缺点:数组的长度是固定的(C99之前),而且数组不提供获取其容量大小的size 操作,也不提供自动添加元素的push_back 操作。因此,程序员无法在程序运行时知道一个给定数组的长度,而且如果需要更改数组的长度,程序员只能创建一个更大的新数组,然后把原数组的所有元素复制到新数组的存储空间中去。与使用vector 类型的程序相比,使用内置数组的程序更容易出错且难以调试。

问题6、注意数组下标越界问题

类似vector容器,string类型,别无他法,一定自习检查。常见的“缓冲区溢出错误”,就是在编写代码的时候,引用了越界的下标,没有检查下标的范围,导致出错!

 

欢迎关注

 

dashuai的博客是终身学习践行者,大厂程序员,且专注于工作经验、学习笔记的分享和日常吐槽,包括但不限于互联网行业,附带分享一些PDF电子书,资料,帮忙内推,欢迎拍砖!

 

 

 

 

posted @ 2014-12-12 01:22  dashuai的博客  阅读(2303)  评论(4编辑  收藏  举报
Flag Counter欢迎关注微信公众号