Chinaunix首页 | 论坛 | 博客
  • 博客访问: 18689572
  • 博文数量: 7460
  • 博客积分: 10434
  • 博客等级: 上将
  • 技术积分: 78178
  • 用 户 组: 普通用户
  • 注册时间: 2008-03-02 22:54
文章分类

全部博文(7460)

文章存档

2011年(1)

2009年(669)

2008年(6790)

分类: C/C++

2008-03-20 18:42:07

来源: 作者:antigloss 等级:强烈推荐
发布于2006-04-01 13:28 被读4050次 【字体: 】

1. 指派初始值(Designated Initializers)

    指派初始值这个特性是 C99 增加的,它允许我们直接初始化数组中特定的元素。C99 以前,如果我们要初始化数组中的某个元素,如第三个元素,必须同时初始化它之前的元素。例如:

        int iarr[10] = { 0, 0, 300 };

而 C99 中,我们可以这样初始化特定的元素:

        int iarr[10] = { [2] = 300 };  /* 指派初始化 iarr[2] 为 300 */

其余的元素都会被初始化为 0 。下面我们来看一个小程序。

        #include <stdio.h>

        int main(void)
        {
            int iarr[5] = { 6, 3, [3] = 1, 5, [1] = 8};
    
            printf("%d\n", iarr[0]);
            printf("%d\n", iarr[1]);
            printf("%d\n", iarr[2]);
            printf("%d\n", iarr[3]);
            printf("%d\n", iarr[4]);

            return 0;
        }

输出为:

        6
        8
        0
        1
        5

从中可以看出两点:

    A. 如果指派初始值后面还有值,则后面的值会被用于初始化后续的元素。上例中,
       iarr[3] 被初始化为 1 ,它后续的元素 iarr[4] 被初始化为 5。

    B. 如果初始化列表中多次出现对某元素的初始化,则以最后一次为准。上例中,
       iarr[1] 先被初始化为 3,然后被 [1] = 8 指派初始化为 8。


2. 给数组元素赋值

    我们可以利用下标给特定的元素赋值。例如:

        int iarr[5];
        iarr[0] = 100;  /* 赋值给第一个元素 */
        iarr[4] = 120;  /* 赋值给第五个元素 */
        iarr[2] = 180;  /* 赋值给第三个元素 */

C 不允许直接使用数组对别的数组进行赋值,也不允许使用初始化列表对数组进行赋值。例如:

        int iarr_1[5] = { 1, 2, 3, 4, 5 };   /* 正确 */
        int iarr_2[5];

        iarr_2 = iarr_1;                     /* 错误! */
        iarr_2[5] = { 3, 4, 5, 6, 7 };       /* 错误! */
        iarr_2[5] = iarr_1[5];               /* 越界! */

最后一个语句发生了越界!因为这两个数组都只有 5 个元素,而使用下标 5 访问的是第六个元素!

 

3. 数组界限(array bounds)

    使用下标时,我们必须确保下标没有越界。例如:

        int iarr[46];

这个数组的下标范围是 0 到 45,确保下标没有超出这个范围是我们的责任,因为编译器不会对下标越界进行检测!

    C 标准没有定义下标越界的后果,也就是说,当我们写的程序中出现下标越界的问题,程序可能正常工作,也可能异常退出,还有可能出现其它奇怪的情况。

        #include <stdio.h>

        int main(void)
        {
            int var_1 = 20;
            int arr[5];
            int var_2 = 40;

            printf("var_1: %d, var_2: %d\n", var_1, var_2);

            arr[-1] = -1;
            arr[5]  =  5;

            printf("%d  %d\n", arr[-1], arr[5]);
            printf("var_1: %d, var_2: %d\n", var_1, var_2);

            return 0;
        }

上述程序使用 Dev-C++ 4.9.9.2 编译运行的输出为:

        var_1: 20, var_2: 40
        -1  5
        var_1: 20, var_2: -1

可见,下标越界可能改变其它变量的值。这是因为 gcc(dev-c++ 使用的 C 编译器)把 var_2 保存于数组 arr 之前的内存空间,所以对 arr[-1] 赋值正好改变了 var_2 的值。不同的编译器编译运行该程序可能会有不同的输出,也可能会异常退出。

    C 语言的哲学是信任程序员,而且不检测越界程序运行更快。程序编译时有些下标的值仍然是不可知的,所以如果要检测下标越界的话,编译器必须在生成的目标代码中加入额外的代码用于程序运行时检测下标是否越界,这就会导致程序运行速度下降。故而,为了运行效率,C 不检测下标是否越界。


4. 指定数组元素数目

    C99 之前,声明数组时,[] 中的值必须是大于零的整数常量。C99 中,声明数组时,[] 中可以是变量。这就是所谓的变长数组(variable-length array,简称 VLA)。声明 VLA 时,不能对其进行初始化。在后续的教程中,我会对 VLA 进行详细讲解。

        int n = 99;
        double dbl_1[4];          /* 正确 */
        double dbl_2[8/2 + 4];    /* 正确 */
        int    iar_1[-5];         /* 错![] 中的值必须大于 0 */
        int    iar_2[0];          /* 错![] 中的值必须大于 0 */
        int    iar_3[9.2];        /* 错![] 中的值必须是整数类型 */
        char   ch[n];             /* C99 之前不支持! */

参考资料:C Primer 5th Edition
          The C Programming Language 2nd Edition
          C99 标准

本文版权归 以及 作者 antigloss 共同所有,转载请注明原作者和出处。谢谢。

阅读(890) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~