有关【指针运算】的经典笔试题

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 有关【指针运算】的经典笔试题

题目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;
}
//程序的输出结果是什么?  2  5

解析:

代码的内存布局如图:

- 考点:

  1. 理解数组名与&数组名的区别。
    数组名一般是指数组首元素地址(除sizeof(数组名)外);&数组名取出的是整个数组的地址。两者的值相同,意义不一样。
  2. 理解指针±整数操作。
    指针±整数的步幅是多少,取决于指针的类型,如整型指针一次解引用访问4个字节,字符指针一次解引用访问1个字节。

题目2:

#include <stdio.h>
//在X86环境下 
//假设结构体的⼤⼩是20个字节 
//程序输出的结果是啥? 
struct Test
{
  int Num;
  char* pcName;
  short sDate;
  char cha[2];
  short sBa[4];
}* p  = (struct Test*)0x100000;
int main()
{
  printf("%p\n", p + 0x1);
  printf("%p\n", (unsigned long)p + 0x1);
  printf("%p\n", (unsigned int*)p + 0x1);
  return 0;
}

解析:

代码解析如图:

所以,第一个printf语句中,p+0x1就等价于指针±整数操作,步幅取决于指针类型的大小,由题可知结构体大小为20个字节,所以结果为0x100000+14(20要转换为十六进制)=0x100014。
第二个printf语句中,p是无符号整型,就是整数,整数+1,结果就是0x100001。
第三个printf语句中,p是整型指针,+1跳过4个字节,结果为0x100004。

- 考点:指针±整数。指针±整数的步幅是多少,取决于指针的类型。

题目3:

#include <stdio.h>
int main()
{
  int a[3][2] = { (0, 1), (2, 3), (4, 5) };
  
  int* p;
  p = a[0];
  printf("%d", p[0]);  //1
  
  return 0;
}

解析:

代码内存布局如图:

这道题也是考查对数组名的理解,a[0]是二维数组第一行的数组名,而数组名又是数组首元素地址,所以p[0]=1。

- 考点:对二维数组的数组名的理解。

题目4:

//假设环境是x86环境,程序输出的结果是啥? 
#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;
}

解析:

代码的解析和内存布局如图:

|指针-指针| = 两者之间元素的个数。由于数组在内存中是由低地址向高地址存储的,由图可知,两个地址的差是-4,%d打印有符号整型,结果为-4;但是%p是以十六进制打印补码,-4的原码为10000000000000000000000000000100,所以补码为11111111111111111111111111111100,转换为十六进制为FFFFFFFC。

题目5:

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

解析:

代码内存布局如图:

&aa取出的是整个二维数组的地址,+1跳过整个二维数组;而aa要理解为二维数组第一行的地址,+1跳过一整行。

题目6:

#include <stdio.h>
int main()
{
 char *a[] = {"work","at","alibaba"};
 
 char**pa = a;
 pa++;
 printf("%s\n", *pa);// at
 
 return 0;
}

解析:

代码的内存布局如图:

这题主要考察对指针数组的理解和二级指针的理解。
指针数组是数组,里面的元素是地址,二级指针存放的是一级指针的地址。

题目7:

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

解析:

代码的内存布局如图:

这个题较为复杂,一定要正确的画出内存布局。并且一定要注意"++cpp和–cpp"这种操作是有"副作用"的运算,它会改变本身的内容!

目录
相关文章
|
7月前
|
机器学习/深度学习 搜索推荐 算法
【再识C进阶2(下)】详细介绍指针的进阶——利用冒泡排序算法模拟实现qsort函数,以及一下习题和指针笔试题
【再识C进阶2(下)】详细介绍指针的进阶——利用冒泡排序算法模拟实现qsort函数,以及一下习题和指针笔试题
|
3月前
|
人工智能
魔法指针 之 指针变量的意义 指针运算
魔法指针 之 指针变量的意义 指针运算
28 0
指针与数组笔试题解析
指针与数组笔试题解析
|
8月前
|
C语言
C语言进阶⑫(指针下)(指针和数组笔试题解析)(杨氏矩阵)(上)
C语言进阶⑫(指针下)(指针和数组笔试题解析)(杨氏矩阵)
55 0
|
8月前
|
C语言
在引用数组元素时指针的运算
在引用数组元素时指针的运算
61 0
|
8月前
指针笔试题模拟
指针笔试题模拟
34 0
|
8月前
进阶指针笔试题
进阶指针笔试题
38 0
|
8月前
|
算法 C语言
C语言进阶⑫(指针下)(指针和数组笔试题解析)(杨氏矩阵)(下)
C语言进阶⑫(指针下)(指针和数组笔试题解析)(杨氏矩阵)
38 0
|
8月前
|
C语言
C语言进阶⑫(指针下)(指针和数组笔试题解析)(杨氏矩阵)(中)
C语言进阶⑫(指针下)(指针和数组笔试题解析)(杨氏矩阵)
43 0
|
8月前
|
存储 编译器 C语言
C语言初阶⑦(指针初阶)知识点+笔试题(下)
C语言初阶⑦(指针初阶)知识点+笔试题
51 0