八道指针笔试题集合,练会让你不再害怕“晕针”

目录

笔试题1

笔试题2

笔试题3

笔试题4 

笔试题5 

笔试题6

笔试题7 

笔试题8


下面代码打印结果是什么? 

笔试题1

#include<stdio.h>

int main()
{
    int a[5] = { 1, 2, 3, 4, 5 };
    int* ptr = (int*)(&a + 1);
    printf("%d,%d", *(a + 1), *(ptr - 1));
    return 0;
}

 解析代码

#include<stdio.h>

int main()
{
    int a[5] = { 1, 2, 3, 4, 5 };
    int* ptr = (int*)(&a + 1);
    //&a取的是整个数组的地址
    //&a+1表示的是数组a[5]元素中5的后面的地址
    //强制转换为int* 类型,使ptr的步伐变为int型大小
    printf("%d,%d", *(a + 1), *(ptr - 1));//2,5
    return 0;
}

笔试题2

#include<stdio.h>
//此结构体的大小是20个字节
struct Test
{
	int Num;
	char* pcName;
	short sDate;
	char cha[2];
	short sBa[4];
}*p;

int main()
{
	p = (struct Test*)0x100000;//假设p的值为0x100000。
	printf("%p\n", p + 0x1);
	printf("%p\n", (unsigned long)p + 0x1);
	printf("%p\n", (unsigned int*)p + 0x1);
	return 0;
}

解析代码

#include<stdio.h>
//结构体的大小是20个字节
struct Test
{
	int Num;
	char* pcName;
	short sDate;
	char cha[2];
	short sBa[4];
}*p;

int main()
{
	p = (struct Test*)0x100000;//假设p的值为0x100000
	printf("%p\n", p + 0x1);//00100014  
    //一个struct Test*类型指针表示的大小为20字节 - +1就是+20个字节
	printf("%p\n", (unsigned long)p + 0x1);//00100001  
    //强制转化为整型,整型+1就是+1个字节 - 用完%p打印的是地址,所以这里会有个格式字符串与可变参数不符的警告
	printf("%p\n", (unsigned int*)p + 0x1);//00100004/00100008   
    //强制转化为int*类型,+1就是+4/8个字节
	return 0;
}

笔试题3

#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);
    return 0;
}

解析代码

#include<stdio.h>

int main()
{
    int a[4] = { 1, 2, 3, 4 };
    int* ptr1 = (int*)(&a + 1);
    int* ptr2 = (int*)((int)a + 1);
    //数组在内存中存储方式:
    //低地址->高地址  小端存储
    //10 00 00 00 02 00 00 00 03 00 00 00 04 00 00 00
    //a表示数组首元素(10 00 00 00)地址,被强制转化为int类型后,再+1只+1个字节
    //即(int)a+1表示10后面的00的地址
    //(int)a+1又被强制转化为int*类型后
    //(int*)((int)a + 1)即ptr2指针访问的大小变为4个字节
    //访问的就是00 00 00 02这块
    //*ptr2打印出来便是02 00 00 00
    printf("%x,%x", ptr1[-1], *ptr2);//4,2000000   
    //ptr1[-1] - *(ptr1+(-1)) - *(ptr1-1)
    
    return 0;
}

笔试题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;
}

解析代码

#include<stdio.h>

int main()
{
    int a[3][2] = { (0, 1), (2, 3), (4, 5) };//小括号 逗号表达式   只取最后的数 - { 1,3,5}
    int* p;
    p = a[0];
    printf("%d", p[0]);//1
    //p = a[0] -> p[0] == a[0][0]
    return 0;
}

笔试题5 

#include<stdio.h>

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;
}

解析代码

#include<stdio.h>

int main()
{
    int a[5][5];
    int(*p)[4];
    p = a;//a - 第一行数组地址  a的类型int(*)[5]   p的类型int(*)[4]   会有警告
    printf("%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);//FFFFFFFC,-4
    //p[4][2] - *(*(p+4)+2)
    //&a[4][2]为红色部分      &p[4][2]为绿色部分  
    //&p[4][2] - &a[4][2] - 指针减指针的数值大小 = 中间元素的个数
    //%d - 打印原码 - -4
    //%p - 直接打印补码
    //10000000000000000000000000000100 - 原码
    //11111111111111111111111111111011 - 反码
    //11111111111111111111111111111100 - 补码
    //0XFFFFFFFC
    return 0;
}

 笔试题6

#include<stdio.h>

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;
}

代码解析

#include<stdio.h>

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));//*(aa+1) - aa[1] - 第二行的数组名 - 第二行首元素地址
    printf("%d,%d", *(ptr1 - 1), *(ptr2 - 1));//10,5
    return 0;
}

笔试题7 

#include<stdio.h>

int main()
{
	char* a[] = { "work","at","alibaba" };
	char** pa = a;
	pa++;
	printf("%s\n", *pa);
	return 0;
}

 代码解析

#include<stdio.h>

int main()
{
	char* a[] = { "work","at","alibaba" };//字符指针数组  放进去的是字符串首字母的地址
	char** pa = a;//字符指针数组指针  pa - 数组名a
	pa++;//a+1
	printf("%s\n", *pa);//at   *pa -> *(a+1) -> a[1]
	return 0;
}

笔试题8

#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);
	printf("%s\n", *-- * ++cpp + 3);
	printf("%s\n", *cpp[-2] + 3);
	printf("%s\n", cpp[-1][-1] + 1);
	return 0;
}

代码解析

#include<stdio.h>

int main()
{
	char* c[] = { "ENTER","NEW","POINT","FIRST" };//字符指针数组
	char** cp[] = { c + 3,c + 2,c + 1,c };
	char*** cpp = cp;
    //运算优先级
    //++、--高于*高于+、-
	//注意:cpp是变量,会随着运算变化,为方便理解,解析中cpp0表示的是初始的cpp含义

	printf("%s\n", **++cpp);//POINT  
    //**++cpp -> **(cpp0+1) -> **(cp+1) -> *cp[1] -> *(c+2) -> c[2] -> POINT

	printf("%s\n", *-- * ++cpp + 3);//ER  
    //*-- * ++cpp + 3 -> *--*(cpp0+2)+3 -> *--cp[2]+3 -> *--(c+1)+3 -> *c+3 -> c[0]+3 -> ENTER首字母地址+3 -> ER

	printf("%s\n", *cpp[-2] + 3);//ST  
    //*cpp[-2] + 3 -> **(cpp0+2-2)+3 -> **cpp0+3 -> *cp[0]+3 -> *(c+3)+3 -> c[3]+3 -> FIRST首字母地址+3 -> ST

	printf("%s\n", cpp[-1][-1] + 1);//EW  
    //cpp[-1][-1] + 1 -> (cpp0+2)[-1][-1]+1 -> (cp+2)[-1][-1]+1 -> *((cp+2)-1)[-1]+1 -> 
    //cp[1][-1]+1 -> (c+2)[-1]+1 -> *((c+2)-1)+1 -> c[1]+1 -> NEW首字母地址+1 ->EW

	return 0;
}
  • 13
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 5
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

天青i

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值