【C语言】指针题及解析


需要云服务器等云产品来学习Linux的同学可以移步/-->腾讯云<--/-->阿里云<--/-->华为云<--/官网,轻量型云服务器低至112元/年,新用户首次下单享超低折扣。


 例一:需要考虑大小端问题(该例仅限32位平台、小端存储模式)

#include <stdio.h>
int main()
{
    int a[4] = { 1, 2, 3, 4 };
    int* ptr1 = (int*)(&a + 1);
    int* ptr2 = (int*)((int)a + 1);
    printf("%x,%x", ptr1[-1], *ptr2);//4,02000000
    return 0;
}

ptr2的解释:a是地址,强转为int,再加1,再强转为int*。整体效果为从a的地址开始,往后跳过一个字节。

ptr2为int*类型,解引用访问四个字节。

小端存储,倒着存,倒着拿。如下图:

例二:指针相减,差值为指针和指针之间的元素个数

#include <stdio.h>
int main()
{
    int a[5][5];
    int(*p)[4];
    p = (int(*)[4])a;
    printf("%p,%d\n",&p[4][2] -&a[4][2],&p[4][2] -&a[4][2]);//FFFFFFFC,-4
    return 0;
}

&p[4][2] -&a[4][2]的差值为-4,以%p的形式打印,得到FFFFFFFC

例三:字符串数组

#include <stdio.h>
int main()
{
	char* a[] = { "hello","world","!!!!" };
	char** pa = a;
	pa++;
	printf("%s\n", *pa);//world
	return 0;
}

a是字符串数组,类型是char*,pa是指向a的指针,pa++,跳过一个元素,解引用即可找到a数组第二个元素的地址,以%s打印,即为world。

例四:多重指针

#include <stdio.h>
int main()
{
	char* c[] = { "ENTER","NEW","POINT","FIRST" };
	char** cp[] = { c + 3,c + 2,c + 1,c };
	char*** cpp = cp;

	printf("%s\n", **++cpp);//POINT
	printf("%s\n", *-- * ++cpp + 3);//ER
	printf("%s\n", *cpp[-2] + 3);//ST
	printf("%s\n", cpp[-1][-1] + 1);//EW

	return 0;
}

按照题目画出图像:

printf("%s\n", **++cpp),首先cpp的地址+1,再连续两次解引用,第一次解引用找到c+1,再次解引用找到c[3],即"POINT"的地址,以%s打印即为POINT。如下图:

printf("%s\n", *-- * ++cpp + 3),首先cpp的地址+1,再解引用找到的是cp[3],再将cp[3]中的地址-1,再解引用找到c[0],再加3表示指向ENTER的指针右移三个字节,即为ER。如下图:

printf("%s\n", *cpp[-2] + 3),首先cpp[-2]可以看成*(cpp-2),即cp[0],再次解引用,即为指向FIRST的指针,再加3,变为指向S的指针,以%s打印,即为ST,如下图:

printf("%s\n", cpp[-1][-1] + 1),可以转化为*(*(cpp-1)-1)+1,即为EW,注意此处的cpp和cp[1]的值是不改变的。如下图:

评论 36
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

蒋灵瑜

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值