需要云服务器等云产品来学习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]的值是不改变的。如下图: