数组和指针

本文深入探讨了C语言中数组地址操作与指针的关系,通过具体实例解释了如何理解和运用数组名、数组地址及指针进行地址运算,揭示了在特定情况下程序输出结果的原因。

先上题:

int main()
{
    int a[5] = {1,2,3,4,5};
    int *ptr = (int *)(&a +1);
    printf("%d", *(ptr-1));
    return 0;
}
对于这道题,我的第一感觉是输出1,但是实际是输出5,为什么呢???

加一条打印语句,代码如下:

int main()
{
    int a[5] = {1,2,3,4,5};
    int *ptr = (int *)(&a +1);
    printf("%X,%X\n",&a,&a+1);
    printf("%d\n", *(ptr-1));
    return 0;
}
输出&a和&a+1,就可以知道当前ptr实际指向a[5]的位置(数组a实际只有5个数)而不是a[0]位置。我们可以打印看看:

#include <stdio.h>
#include <stdlib.h>

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

上一个程序打印*ptr是乱码,下一段代码打印ptr是100。建议将下面程序的数组a设置成全局变量,而不是局部变量。因为局部变量是在栈里面的,栈用于保存函数的局部变量和函数调用的环境变量,所以对栈内的没分配的空间乱调用也许会造成程序运行混乱。但是全局变量是在堆里分配的。

#include <stdio.h>
#include <stdlib.h>
int a[5] = {1,2,3,4,5};
int main()
{
    int *ptr = (int *)(&a +1);
    a[5]=100;
    printf("%d\n",*ptr);
    printf("%d", *(ptr-1));
    return 0;
}
这道题的关键在于理清&a的类型是in*,还是int*[5]。
下面还有几个类似的:

#include <stdio.h>
#include <stdlib.h>

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

#include <stdio.h>
#include <stdlib.h>

typedef int y[5];

int main()
{
    int a[5] = {1,2,3,4,5};
    y *ptr = &a + 1;
    printf("%X\n",a);
    printf("%X\n",&a);
    printf("%X\n",&a+1);
    printf("%X\n",ptr);
    printf("%X\n",ptr-1);
    printf("%d\n", *((int*)(ptr-1)));
    printf("%X\n",&a);
    printf("%X\n",*(&a));
    return 0;
} 
#include <stdio.h>
#include <stdlib.h>

typedef int y[5];

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

#include <stdio.h>
#include <stdlib.h>

typedef int y[5];

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




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值