作者: 华丞臧. 专栏:【C语言】 各位读者老爷如果觉得博主写的不错,请诸位多多支持(点赞+收藏+关注)。如果有错误的地方,欢迎在评论区指出。
笔试题1
int main()
{
int a[5] = { 1, 2, 3, 4, 5 };
int *ptr = (int *)(&a + 1);
printf( "%d,%d", *(a + 1), *(ptr - 1));
return 0;
}
程序的结果是2,5。 需要注意的是&a 取出的是整个数组的地址,&a+1 表示跳过整个数组的地址指向数组后面的空间。
笔试题2
struct Test
{
int Num;
char *pcName;
short sDate;
char cha[2];
short sBa[4];
}*p;
int main()
{
printf("%p\n", p + 0x1);
printf("%p\n", (unsigned long)p + 0x1);
printf("%p\n", (unsigned int*)p + 0x1);
return 0;
}
程序结果是0x00 00 00 14,0x00 00 00 01,0x00 00 00 04。 已知结构体大小为20个字节,p 是结构体指针那么p+1 就表示跳过一个结构体大小的地址指向其后面的地址。
笔试题3
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);
return 0;
}
程序的结果是4,2000000。 显然ptr1 是跳过a 数组指向a 数组后面的地址,ptr1[-1] 相当于*(ptr1-1) 所以ptr1[-1] 会指向a 数组最后一个元素4 。
ptr2 = (int *)((int)a + 1);
我的系统是小端字节序存储,根据小端存储数据的方式,a 数组首元素在内存中地址从低到高存储的是 01 00 00 00 ,可以知道*ptr2 中存储的是 02 00 00 00 ,
笔试题4
#include <stdio.h>
int main()
{
int a[3][2] = { (0, 1), (2, 3), (4, 5) };
int *p;
p = a[0];
printf( "%d", p[0]);
return 0;
}
程序结果是1. p 中存放的是a 数组第一行的地址 p[0] 等价于 p[0][0] ,也就是第一行第一个元素;a[0][0] = = 0。逗号表达式的值等于逗号表达式中最后一个式子的值。
笔试题5
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;
}
程序的结果是0x ff ff ff fc ,-4。
从下图可以看出,&p[4][2] - &a[4][2] = -4 。
-4的二进制原码为:1000 0000 0000 0000 0000 0000 0000 0100 -4的二进制反码为:1111 1111 1111 1111 1111 1111 1111 1011 -4的二进制补码为:1111 1111 1111 1111 1111 1111 1111 1100 内存中存放的是补码,而&p是以地址的形式打印,所以打印的是: 二进制:1111 1111 1111 1111 1111 1111 1111 1100 16进制:ff ff ff fc
笔试题6
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。
(&aa+1) 表示跳过整个数组指向数组地址后面的地址。 *(aa+1) 取出第二行的地址,第二行的地址就是第二行首元素的地址即a[1][0] 的地址。
笔试题7
#include <stdio.h>
int main()
{
char *a[] = {"work","at","alibaba"};
char**pa = a;
pa++;
printf("%s\n", *pa);
return 0;
}
程序结果是"at "。 pa 是字符指针,那么我们要知道字符指针数组a 当中存储字符串时存储的是字符串中首字母的地址;所以当pa 指向a 时,pa++ 跳过的是一个一级字符指针,此时pa 中存储的是a[1] 的地址,*pa == a[1] == "at" 。
笔试题8
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;
}
程序结果是POINT、ER、ST、EW。
注意:
- 指针后面加
[N] 表示指针加上N 再解引用。 - 前置和后置 加减 会永久改变指针变量的值。
int arr[] = {1,2,3,4,5];
int* p = arr;
解析:
|