C提高——数组、数组数据类型、数组数据类型指针、多维数组本质等

数组

int main() {
    int a[200] = { 0 };


    printf("a=%d\n", a);
    printf("&a=%d\n", &a);

    printf("a+1=%d\n", a+1);
    printf("&a+1=%d\n", &a+1);
    system("pause");
}

这里写图片描述

对于一维数组,C语言规定:
a代表数组首元素的地址
&a代表整个数组的地址
数组的地址和数组首元素的地址是重叠的。

但两个指针的步长不一样,a+1的步长为4,&a+1的步长为200*4

数组(数据)类型

怎样定义一个指向数组 的 数据类型?
typedef int(MYARRAY)[5];// 定义了一个数组数据类型,相当于 int array[10];

    typedef int(MYARRAY)[5];// 定义了一个数组数据类型,相当于 int array[10];
    MYARRAY array;
    int len = sizeof(array);
    memset(array, 0, len);

    array[0] = 10;
    array[1] = 20;

    int i;
    for (i = 0; i<len / 4; i++){
        printf("%d\n", array[i]);
    }

    printf("array=%d\n", array);
    printf("&array=%d\n", &array);
    printf("array + 1=%d\n", array + 1);
    printf("&array + 1=%d\n", &array + 1);

    system("pause");
}

这里写图片描述

同上面数组一样:
array代表数组首元素的地址
&array代表整个数组的地址

数组指针(数据)类型

怎样定义一个指向数组指针的数据类型?
同数组(数据)类型类似:
typedef int(*pArray)[5];// 定义一个数组指针类型

数据指针类型变量

定义数组指针类型变量的三种方法:
一、通过声明的数组类型定义数组指针变量:

int main() {
    typedef int (MyArray)[5];// 声明一个数组类型
    MyArray *p;// 通过数组类型 定义 数组指针变量

    int arr[5] = { 0 };
    p=&arr;
    int i;

    for (i = 0; i < 5;i++){
        (*p)[i] = i + 10;
    }

    for (i = 0; i < 5; i++){
        printf("%d\n", (*p)[i]);
    }
    system("pause");
}

这里写图片描述

二、通过声明的数组指针类型定义变量:

int main() {
    typedef int (*pArray)[5];// 声明一个数组指针类型
    pArray p;// 定义一个数组指针类型变量,告诉编译器分配4个字节的内存(32位平台下)

    int arr[5] = { 0 };// 一维数组名相当于一级指针
    p=&arr;// 数组指针变量相当于一个二级指针
    int i;

    for (i = 0; i < 5;i++){
        (*p)[i] = i + 10;
    }

    for (i = 0; i < 5; i++){
        printf("%d\n", (*p)[i]);
    }
    system("pause");
}

这里写图片描述

三、直接定义

int main() {
    int (*p)[5];// 直接定义数组指针类型变量,,告诉编译器分配4个字节的内存(32位平台下)

    int arr[5] = { 0 };
    p=&arr;
    int i;

    for (i = 0; i < 5;i++){
        (*p)[i] = i + 10;
    }

    for (i = 0; i < 5; i++){
        printf("%d\n", (*p)[i]);
    }
    system("pause");
}

这里写图片描述

多维数组名的本质

int main() {

    int a[3][5];
    int i, j;
    int temp = 1;

    for (i=0; i < 3; i++) {
        for (j=0; j < 5; j++) {
            a[i][j] = temp++;
        }
    }

    // 直接使用二维数组名
    for (i = 0; i < 3; i++) {
        for (j = 0; j < 5; j++) {
            printf("%d ", a[i][j]);
        }
    }

    printf("\n");

    int(*p)[5];// 定义一个指向一维数组的指针变量
    p = a;
    // 使用指针变量
    for (i=0; i < 3; i++) {
        for (j=0; j < 5; j++) {
            printf("%d ", p[i][j]);
        }
    }

    printf("\n");

    printf("*a=%d\n", *a);
    printf("a=%d\n", a);
    printf("a+1=%d\n", a + 1);
    printf("&a=%d\n", &a);
    printf("&a+1=%d\n", &a + 1);

    printf("p=%d\n", p);
    printf("p+1=%d\n", p+1);

    system("pause");
}

这里写图片描述

从上图结果可知:
二维数组的地址&a、二维数组首行的地址a、二维数组首元素的地址*a的值是重合的。
a+1的步长是20,刚好等于第一行的长度5*4。指向二维数组的数组指针p的步长也是20,证明二维数组名相当于一维数组指针
&a+1的步长是60,刚好是整个二维数据的长度15*4。

推演:
(a+i) 代表整个第i行的地址
*(a+i) 代表第i行首元素的地址
*(a+i)+j 代表第i行,第j列的地址,相当于&a[i][j]

这里写图片描述

总结:(多维数组名 a)的本质是 (一个指向一维数组的指针),a的步长就是单个行的长度

数组 “[]” 和 指针“*”的转化推演

a[i] = *(a+i)的转化过程:
a[i]
—-> a[0 + i]: 第一步,加上0占位符
—->[a + i]:第二步用a替换占位符0
—-> *(a + i):第三步将中括号改为小括号,并在前面加上*

a[i][j] = *( *(a + i) + j)的转化过程:
a[i][j]
—->a[0 + i][j]:第一步,加上0占位符
—->[a + i][j]:第二步,用a替换占位符0
—-> *(a + i)[j]:第三步,将中括号改为小括号,并在前面加上*
—->*(a + i)[0 + j]:第四步,加上0占位符
—->[*(a + i) + j]:第五步,用*(a + i)替换占位符0
—->*( *(a + i) + j):第六步,将中括号改为小括号,并在前面加上*

二维数组做函数参数的退化问题

void printArr(int(*p)[5]) {
    int i, j;
    for (i = 0; i < 3; i++) {
        for (j = 0; j < 5; j++) {
            printf("%d ", p[i][j]);
        }
    }
}

int main() {

    int a[3][5];
    int i, j;
    int temp = 1;

    for (i=0; i < 3; i++) {
        for (j=0; j < 5; j++) {
            a[i][j] = temp++;
        }
    }

    printArr(a);

    system("pause");
}

这里写图片描述

多维数组做函数参数会退化为一个指向一维数组的指针。也就是说形参中的多维数组编译器会把它当成一个指向一维数组的指针处理。实参a和形参p的数据类型本质不一样。

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值