指针进阶习题

本文探讨C语言中指针与数组的关系,详细解释了sizeof运算符在数组和指针上的应用,以及二维数组和逗号表达式中指针的操作。通过实例分析了指针加减运算的含义,强调了实践对于理解和掌握指针的重要性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

int main()
{
	int a[] = { 1,2,3,4 };
	int* p1 = (int*)(&a + 1);//这里取地址a,是取出整个数组的地址,整个数组地址+1,跳过整个数组,
	int* p2 = (int*)((int)a + 1);
	printf("%x,%x", p1[-1], *p2);
	return 0;
}

这里取地址a,是取出整个数组的地址,整个数组地址+1,跳过整个数组,然后强制转化为int*,其实也就是int(*p)[10],指向数组的指针-1,向后跳过一个整形,解引用得到4。

这里总结以下sizeof(arr)和sizeof(&arr)有所不同,sizeof(arr)相当于计算了里面元素个数,而sizieof(&arr)取出的是地址,可以理解为指针,而指针就是4/8个字节。取出数组的地址是要放在数组指针其实取出的也就是int(*p)【10】。而sizeof(&arr + 1))就是取出整数组的地址,整个数组地址+1还是一个地址,所以就为4/8个字节。

 

以上比较容易迷惑的是sizeof(p+1)),因为p是个指针,指针+1向下跳过一个字节,其实就是b的地址,也就是4个字节,而sizeof(*p))对p指针指向额内容解引用就是访问到里面的元素了,字符大小也就是1个字节。

二维数组

int main()
{
	int a[3][4] = { 0 };
	printf("%d\n", sizeof(a));//48
	printf("%d\n", sizeof(a[0][0]));//4
	printf("%d\n", sizeof(a[0]));//16
	printf("%d\n", sizeof(a[0] + 1));//4
	printf("%d\n", sizeof(*(a[0] + 1)));//4
	printf("%d\n", sizeof(a + 1));//4
	printf("%d\n", sizeof(*(a + 1)));//16
	printf("%d\n", sizeof(&a[0] + 1));//4
	printf("%d\n", sizeof(*(&a[0] + 1)));//16
	printf("%d\n", sizeof(*a));//16
	printf("%d\n", sizeof(a[3]));//16
	return 0;
}

二维数组比较容易迷惑的一点是printf("%d\n", sizeof(a[0] + 1));//4  。以为这里a[0]不是单独放在操作符内部,所以表示的是首元素地址,首元素地址+1跳过一个整形。而非整个数组或者一个一维数组。    printf("%d\n", sizeof(*(a[0] + 1)));//4这个计算的是元素的大小,而上面那个计算的是指针的大小。    printf("%d\n", sizeof(a + 1));//4这里(a+1)表示的表示首元素地址,+1得到第二行地址,其实也就还是指针,大小为4/8个字节。

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

逗号表达式

int main()
{
	int a[3][2] = { (0, 1), (2, 3), (4, 5) };
	int* p;
	p = a[0];
	printf("%d", p[0]);
	return 0;
}

输出结构为1,

important code

int main()
{
	int a[5][5];
	int(*p)[4];
	p = a;
	printf("%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);//FFFFFFFC	-4
	return 0;
}
int main()
{
	int aa[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
	int* ptr1 = (int*)(&aa + 1);
	int* ptr2 = (int*)(*(aa + 1));
	printf("%d,%d", *(ptr1 - 1), *(ptr2 - 1));
	return 0;
}

上面这段代码打印出为10 .5 int* ptr2 = (int*)(*(aa + 1));(aa+1)并没有单独放在括号内部,其实就是首元素地址,也就是一维数组地址,+1得到第二行地址赋给指针2,后面又-1得到5。

int main()
{
	char* a[] = { "work","at","alibaba" };
	char** pa = a;
	pa++;
	printf("%s\n", *pa);
	return 0;
}

关于指针这一节感觉还是要多看代码,然后多动手实践一下不然相当于竹篮打水总是前学后忘。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值