C语言指针相关——经典面试题

目录

1. 指针类型决定指针+ -的步长

2. 注意逗号表达式()与 { } 之间区别

3. 二维数组与指针

4. 字符指针


1. 指针类型决定指针+ -的步长

强制类型转换:类型转换,但是值不会改变;

	int a[5] = { 1, 2, 3, 4, 5 };
	int *ptr = (int *)(&a + 1);//强制类型转换:类型转换,但是值不会改变;
	printf( "%d,%d", *(a + 1), *(ptr - 1));
	//答案:2 5
	//已知,结构体Test类型的变量大小是20个字节
	p = (struct Test*)0x100000;//p就是结构体类型的指针
	printf("%p\n", p + 0x1);//这里指针类型是20(0x14)字节
	printf("%p\n", (unsigned long)p + 0x1);
    //这里把p转换成整数类型,整型+-1就是直接+-1,所以直接加 0x1

	printf("%p\n", (unsigned int*)p + 0x1);//这里指针类型是4(0x4)字节
    //答案:00100014/00100001/00100004
	int a[4] = { 1, 2, 3, 4 };
	int *ptr1 = (int *)(&a + 1);
	int *ptr2 = (int *)((int)a + 1);//***注意***:此时a为整型,那么a+1的值就是a+1
	printf( "%x,%x", ptr1[-1], *ptr2);// %x是以16进制输出整数;
	//注意:ptr1[-1]就是 *(ptr1 - 1)
	//下面给出a数组在内存中的存储(每个字节)
	//01 00 00 00 02 00 00 00 03 00 00 00 04 00 00 00
	//a本来是地址指向01,+1之后(字节+1)指向01后面的00了
	//ptr2指向整个序列的第二个字节,且ptr2是整型指针类型,解引用后访问4个字节:00 00 00 02 (小端实际要反过来)= 0x2000000
	//答案: 4 2000000

2. 注意逗号表达式()与 { } 之间区别

	int a[3][2] = { (0, 1), (2, 3), (4, 5) };
	//注意,里面若是{}括号,那么就是正常解答
	//(,)是逗号表达式,实际上a[3][2]  = {1,3,5}
	int *p;
	p = a[0];//表示第一行首元素的地址
	printf( "%d\n", p[0]);//p[0] == *(p+0),p[0]表示a[0][0],第一行第一个元素
	//答案是1

3. 二维数组与指针

	int a[5][5] = {0};
	int(*p)[4];
	p = (int(*)[4])a;
	printf("%d\n",&p[4][2] - &a[4][2]);//-4  ***重要***:两个地址相减得到是中间元素的个数
	printf("%p\n",&p[4][2] - &a[4][2]);//fffffffc -4以%p的形式打印时,直接将补码以地址的形式读取;
	//以上a[4][2]容易获得
	//p[4][2]的获取,画出a[5][5]在内存中的存储
	//00000 00000 00000 00000 00000
	//p是int(*)[4]型指针,指向a[5][5]第一行第一个元素的地址;
	//p[4]相当于*(p+4),p向后指向了4*4个元素,也就是a[3][1]的位置
	//p[4][2] == (*(p+4))[2] == *((*(p+4))+2)
	//(p+4)是地址,*(p+4)解引用是数组名,且p解引用后*p应该是4个元素的数组,(p+4)也一样
	//所以(*(p+4))[2]向后指2个元素,也就是a[3][3]
	//a[3][3]和a[4][2]中间有4个元素;
	int aa[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
	int *ptr1 = (int *)(&aa + 1);
	int *ptr2 = (int *)(*(aa + 1));//第二个int*其实可以不用
	//(aa+1)表示第二行(是一个一维数组)的地址,解引用*(aa+1)表示第二行的数组名,也表示第二行的首元素的地址
	//int *ptr2 = (*(aa + 1));
	//int *ptr2 = (int *)((aa + 1));
	printf( "%d,%d\n", *(ptr1 - 1), *(ptr2 - 1));
	//答案: 10 5

4. 字符指针

	char *a[] = {"work","at","alibaba"};//字符串数组
	char** pa = a;//a是数组名,pa指向首元素(字符指针)的地址
	pa++;//(pa+1) 指针向后移了一项;
	printf("%s\n", *pa);
	//at
	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
	//讲解见笔记:画一下数组的内存分布再做题;
	//注意:++cpp会对cpp重新赋值,相当于cpp = (cpp+1)
	//而cpp[-2]等价于 *(cpp-2),不会cpp重新赋值

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值