【C语言】指针题及解析

【C语言】指针题及解析_第1张图片


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

#include 
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*类型,解引用访问四个字节。

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

【C语言】指针题及解析_第2张图片

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

#include 
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

【C语言】指针题及解析_第3张图片

例三:字符串数组

#include 
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。

【C语言】指针题及解析_第4张图片

例四:多重指针

#include 
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;
}

按照题目画出图像:

【C语言】指针题及解析_第5张图片

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

【C语言】指针题及解析_第6张图片

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

【C语言】指针题及解析_第7张图片

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

【C语言】指针题及解析_第8张图片

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

【C语言】指针题及解析_第9张图片

你可能感兴趣的:(C语言,c语言,指针,c++)