C语言数组指针问题

在网上看到一个这样代码:

int main
​{
    ​int a[5] = {1,2,3,4,5};
    ​int *str = (int *)(&a+1);
    ​printf("%d\n%d\n", *(a+1), *(str-1));
    ​return 0;
​}​

我一开始也陷入了陷阱里,没想这一个数组指针,以为只是普通的首地址加上一个数据类型的字节,于是我开始进行了下面一系列的实验:

#include<stdio.h>
int main(void)
{
    int a[5] = { 1,2,3,4,5 };
    int *b = (int*)(&a+1);
    for (int count = 0; count < 6; count++)
    {
        printf("&a[%d]=%d\n", count, a + count);
        printf("&a[%d]=%d\n", count, &a[count]);
    }
    printf("\n");
    printf("b=%d\n",b);
    printf("b-1=%d\n", b-1);​
    printf("*b=%d\n",*b);
    printf("*b-1=%d\n",*b-1);
    printf("*(b-1)=%d\n",*(b-1));
    return 0;
}

打印出来的结果如图一所示:
图一

因为指针类型是int*类型,所以在每+1为4个字节(编译环境:VS2015)。b和a[5]的地址相同,其实这个写法是错的,会越界。b-1和a[4]地址相同。所以可以看出这个+1加的&a是整一个数组的长度。

那是不是这样呢?我们把上面的printf(“&a[%d]=%d\n”, count, a + count);改成printf(“&a[%d]=%d\n”, count, &a + count);
打印结果如图二所示:
图二

可以看到&a+1之间是以20个字节为单位,&a[1]之间以4个字节为单位。证明我上面说的,那+1该加多少呢?是不是与前面的地址所指的长度有关?我们把上面的printf(“&a[%d]=%d\n”, count, &a + count);改成printf(“b + %d =%d\n”, count, b + count);,我们来看一下结果:
图三

我们可以看到第一个b+0还是&a[5]的地址,那为什么后面是以4个字节为单位呢?而不是以20个字节呢?那是因为在定义b这个指针的时候,把他的类型转换成了int*型(int b=(int)(&a+1));所以现在可以证明:+1要加多少跟+1前面的指针类型有关系。

那二维数组的时候,情况又是怎么样的呢?我又写了以下的程序进行测试:​

#include<stdio.h>
int main(void)
{
    int a[3][3] = { { 1,2 },{ 3,4 },{ 5,6 } };
    int *b = (int*)(&a + 1);
    for (int c = 0; c < 3; c++)
    for (int d = 0; d < 3; d++)
    printf("&a[%d][%d]=%d\n", c, d, &(a[c][d]));
    printf("\n");
    for (int count = 0; count < 3; count++)
    {
        printf("a + %d=%d\n", count, a + count);
        printf("&a[%d]=%d\n", count, &a[0] + count);
    }
    printf("\n");
    printf("&a[2][1]=%d\n", &(a[2][1]));
    printf("b=%d\n", b);
    printf("b-1=%d\n", b - 1);
    printf("*b=%d\n", *b);
    printf("*b-1=%d\n", *b - 1);
    printf("*(b-1)=%d\n", *(b - 1));
    return 0;
}

结果如图四所示:
图四

我们可以看出a+count是以12个字节为单位,因为我们定义的是int型3*3的二维数组,所以一行为12。在二维数组的指针中a=&a[0][0],a+1=&a[1][0],依此类推。

我们现在把​printf(“a + %d=%d\n”, count, a + count);改成printf(“&a + %d=%d\n”, count, &a + count);看看是不是以3*3*4=36个字节为单位递增,我们打印看一下结果:
图五

看来我们的推导是正确的,果然是以36个字节为单位递增。所以:+1要加多少个字节跟+1前面的指针类型有关系。&a是数组指针,代表数组的总长度,所以&a+1中的1为数组a所占的内存。而a+1中的a跟&a[0][0]是一样的,都表示的是数组的首地址,所以其中+1中的1为数组的类型字节数。
以上实验均在windows平台上运行,Linux平台下结果不同。

<think>嗯,用户之前问过怎么声明一维数组变量,现在又问C语言数组指针类型。看来他们对数组指针的关系感兴趣,特别是在C语言中的具体应用。我需要先确认他们对C语言的基础知识有没有掌握,比如指针的基本概念和数组的声明。 用户可能正在学习C语言,遇到了数组指针相关的问题,或者在做项目时需要用到数组指针。他们可能已经了解了如何声明数组,现在想进一步理解如何用指针来操作数组,或者如何将数组作为参数传递给函数。 C语言中的数组指针关系密切,但这也是容易混淆的地方。用户可能想知道数组指针的具体类型定义,如何声明指向数组指针,以及这样的指针有什么用途。比如,指向一维数组指针和指向数组元素的指针有什么区别,或者如何处理多维数组指针。 需要区分指针数组数组指针的不同。指针数组是一个数组,元素都是指针;而数组指针是一个指针,指向整个数组。例如,int (*p)[5] 是一个指向包含5个int元素的数组指针。这一点需要明确,避免用户混淆。 还要注意数组名在大多数情况下会退化为指向首元素的指针,但在某些情况下,比如使用sizeof时,数组名代表整个数组的大小。这可能涉及到指针运算的不同,比如当指针指向整个数组时,加减操作会跨越整个数组的长度。 用户可能需要具体的例子来理解如何声明和使用数组指针。例如,如何通过数组指针访问数组元素,或者如何将二维数组的行指针传递给函数。这时候需要给出清晰的代码示例,展示声明、初始化和访问的过程。 另外,应用场景也很重要。比如在函数参数中传递多维数组时,数组指针非常有用。或者当处理动态分配的多维数组时,正确使用数组指针可以简化代码。用户可能在实际编程中遇到了相关问题,需要知道如何正确声明和使用这样的指针来解决问题。 还要注意常见的错误,比如将数组指针指针数组混淆,或者在指针运算时步长计算错误。这时候需要强调语法上的区别,比如int *p[5] 和 int (*p)[5]的不同,以及它们各自适用的场景。 总结来说,用户需要的是对C语言数组指针类型的清晰解释,包括定义、声明方法、使用场景以及常见错误的避免。通过结构化的讲解和示例代码,帮助他们理解这一概念,并能够正确应用到实际编程中去。</think>在C语言中,**数组指针**(或称**指向数组指针**)是一种特殊的指针类型,它指向一个完整的数组(而非单个元素)。以下是详细说明: --- ### **1. 数组指针的定义** 数组指针的声明语法为: `数据类型 (*指针名)[数组长度];` 例如: ```c int (*p)[5]; // p是一个指针,指向长度为5的整型数组 ``` --- ### **2. 与普通指针的区别** - **普通指针**指向单个元素: ```c int *p; // p指向一个int类型变量 ``` - **数组指针**指向整个数组: ```c int (*p)[5]; // p指向一个长度为5的整型数组 ``` --- ### **3. 数组指针的赋值** 需将数组的地址赋值给指针数组名默认退化为首元素地址,需用`&`获取整个数组地址): ```c int arr[5] = {1, 2, 3, 4, 5}; int (*p)[5] = &arr; // p指向数组arr ``` --- ### **4. 通过指针访问数组** 通过数组指针访问元素时,需结合`*`和下标: ```c printf("%d", (*p)[0]); // 输出第一个元素:1 printf("%d", (*p)[2]); // 输出第三个元素:3 ``` --- ### **5. 多维数组中的应用** 数组指针常用于处理**二维数组**的行指针: ```c int arr[3][4] = {{1,2,3,4}, {5,6,7,8}, {9,10,11,12}}; int (*p)[4] = arr; // p指向二维数组的第一行 printf("%d", p[1][2]); // 输出第二行第三列元素:7 ``` --- ### **6. 关键特性** - **指针步长**: 若`p`指向长度为5的数组,则`p+1`会跳过整个数组(`5*sizeof(int)`字节)。 - **类型严格匹配**: 数组长度是类型的一部分。例如,`int (*p)[5]`和`int (*p)[6]`是不同类型。 --- ### **7. 常见错误** - **混淆指针类型**: ```c int *p = arr; // p指向arr的首元素(等价于&arr[0]) int (*q)[5] = &arr; // q指向整个数组arr ``` - **误用指针运算**: 操作`q+1`会跳过一个完整的数组,而`p+1`仅跳过一个元素。 --- ### **总结** 数组指针的核心用途是**操作连续内存块**(如多维数组),需注意其类型和步长的特殊性。理解它与普通指针的区别,能更灵活地处理复杂数据结构。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值