既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上C C++开发知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新
1、指针的练习题
1.1 练习 1
int main()
{
int a[5] = { 1, 2, 3, 4, 5 };
int \*p = (int \*)(&a + 1);
printf( "%d,%d", \*(a + 1), \*(p - 1));
return 0;
}
运行结果见下图:
1.2 练习 2
struct Test* p,p是一个结构体类型的指针,结构体Test类型的变量大小是20个字节。
假设p 的值为0x100000。 下面表达式的值分别为多少?
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;
}
分析上面代码:
- p是结构体类型的指针,p+0x01,指针移动1位,跳过整个结构体,共20个字节,16进制为增加14,打印100014
- (unsigned long)p,将p强制转换成无符号整形,p+0x01,就是之间加1,打印100001
- (unsigned int*)p,将p强制转换成无符号整形指针,p+0x01,指针移动1位,指向下一个无符号整形,即移动4个字节,打印100004
结果见下图,VS的默认值不一样,但是看尾数的变化,与分析一致。
1.3 练习 3
int main()
{
int a[4] = { 1, 2, 3, 4 };
int \*p1 = (int \*)(&a + 1);
int \*p2 = (int \*)((int)a + 1);
printf( "%x,%x", p1[-1], \*p2);
return 0;
}
分析上面代码:
- &a整个数组的地址。&a+1,指针跳过整个数组,第一个结果输出4
- (int)将首元素地址转换为整数;(int)+1,就是数值直接加1;(int*)((int)+1),将数值转换成指针,也就是地址了,地址数值比原来增加1,意味着指针移动一个字节。
- *p2 是整形,解引用后访问4个字节 00 00 00 02,这在内存中是由低地址向高地址排列的,小端存储打印出来2000000
输出结果见下图,与分析一致:
1.4 练习 4
int main()
{
int a[3][2] = { (0, 1), (2, 3), (4, 5) };
int \*p;
p = a[0];
printf( "%d", p[0]);
return 0;
}
分析上面代码:
- 二维数组内是小括号,是逗号表达式,所以数组定义见下图
- p[0] = a[0][0] = 1
运行结果见下图,与分析一致:
1.5 练习 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;
}
分析上面代码:
- 数组指针,指向一个数组,这个有4个元素,每个元素都是整形
- p + 1,移动一个数组的长度,就是4个整形
- &p[4][2] - &a[4][2] 就是两个指针相减,结果是指针之间元素的个数,就是4个元素。低地址-高地址,结果是-4
- -4以%d形式打印,就是-4。但是以%p形式打印,就要详细分析:
10000000 00000000 00000000 00000100 -4原码
11111111 11111111 11111111 11111011 -4反码
11111111 11111111 11111111 11111100 -4补码
补码当地址
FF FF FF FC
- 将 -4 的补码认为就是地址,则打印 FF FF FF FC
结果见下图,与分析一致:
1.6 练习 6
int main()
{
int a[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
int \*p1 = (int \*)(&a + 1);//&a是整个数组的地址,&a+1跳过一个数组,即40个字节
int \*p2 = (int \*)(\*(a + 1));//a数组名,不是2中例外之一,表示首元素地址,a+1表示第二个元素的地址
//\*(a+1)解引用拿到第二个元素的内容,即第二行数组,也是第二行数组的数组名
printf( "%d,%d", \*(p1 - 1), \*(p2 - 1));
return 0;
}
分析上面代码,见下图所示:
运行结果见下图,与分析一致:
1.7 练习 7
int main()
{
char \*a[] = {"work","at","alibaba"};
char\*\*pa = a;
pa++;
printf("%s\n", \*pa);
return 0;
}
分析上面的代码:
- 要注意,指针数组里存放的不是整个字符串,而是字符串首元素的地址
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上C C++开发知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新
img-ewluTPcD-1715557821622)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上C C++开发知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新