题一
struct Test
{
int Num;
char *pcName;
short sDate;
char cha[2];
short sBa[4];
}*p;
//假设p 的值为0x100000。 如下表表达式的值分别为多少?
//已知,结构体Test类型的变量大小是20个字节
int main()
{
printf("%p\n", p + 0x1); //0x100014,p指针+1跳过20字节
printf("%p\n", (unsigned long)p + 0x1); //0x100001,强制转换成长整型,+1就是+1,不是跳过一个指针的大小
printf("%p\n", (unsigned int*)p + 0x1); //0x100004,强制转换成无符号整型指针,+1跳过4个字节
return 0;
}
题二
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,2000000
printf( "%#x,%#x", ptr1[-1], *ptr2); //0x4,0x2000000
return 0;
}
//数组a按16进制在内存中进行存储,ptr1[-1]=*(ptr1 - 1);
//a中存储的是首元素地址,转换成整型后+1,再转换成整型指针,ptr2从此位置处向后读4个字节即为00 00 00 02,因为是小端存储,打印的数据为0x 02 00 00 00
//小端存储:比如1《=》00 00 00 01放入内存中变为01 00 00 00,低位先行,低位存低位,高位存高位
题三
int main()
{
int a[3][2] = { (0, 1), (2, 3), (4, 5) }; // 1 3
int *p; // 5 0
p = a[0]; //数字1的地址 // 0 0
printf( "%d", p[0]); //1,p[0]《=》*(p+0)《=》取出数字1地址中存放的内容(1)
return 0;
}
题四
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]);
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;
}
//&aa取出来的是整个二维数组的地址,+1指针指向prt1的位置,再强制转换成整型指针类型,ptr1-1,指针向前移动4个字节大小,指向数字10前的位置,再取其地址,得到10
//*(aa+1)=aa【1】:二维数组第二行首元素地址,保存在ptr2中,ptr2-1指针指向数字5前,取其地址得到5
题六
int main()
{
char *a[] = {"work","at","alibaba"}; //是将三个字符串的首字母地址保存到指针数组中
char **pa = a; //pa中保存指针数组a中首元素地址,即‘w’地址的地址
pa++; //+1后指向数组中第二个元素的地址,即‘a’地址的地址
printf("%s\n", *pa); //at,*pa,从‘a’地址开始以字符串形式打印
return 0;
}
题七
int main()
{
char *c[] = {"ENTER","NEW","POINT","FIRST"};
char**cp[] = {c+3,c+2,c+1,c};
char***cpp = cp;
printf("%s\n", **++cpp);
printf("%s\n", *--*++cpp+3);
printf("%s\n", *cpp[-2]+3);
printf("%s\n", cpp[-1][-1]+1);
return 0;
}