【C语言】进阶指针(三)—>指针与数组笔试真题详解(下)

简介: 【C语言】进阶指针(三)—>指针与数组笔试真题详解(下)

笔试题(2)

struct Test
{
     int Num;
     char *pcName;
     short sDate;
     char cha[2];
     short sBa[4];
}*p;
//假设p 的值为0x100000。 如下表表达式的值分别为多少?
//已知,结构体Test类型的变量大小是20个字节
int main()
{
     printf("%p\n", p + 0x1);
     printf("%p\n", (unsigned long)p + 0x1);
     printf("%p\n", (unsigned int*)p + 0x1);
     return 0;
}

答案:0x100014 0x100001 0x100004

分析:

p的类型为结构体指针struct Test*,步长为struct Test,0x1是十六进制数字,转为十进制为1,+1后相当于+20(整型),转化为十六进制加到0x100000后即为0x100014。

p被转化为数字,+1就是0x100000+1,为0x100001。

p被转化为unsigned int*类型,步长为4个字节,+1就是跳过4个字节,即0x100000+4=0x100004。


笔试题(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+1指针指向元素'4'后面的位置,将该指针强制转换为int*类型,ptr1[-1]相当于*(ptr1-1),又因为ptr1的步长为int,所以解引用得到元素为'4',用%x(十六进制打印)得到值为4。

对于ptr2:

涉及到了内存的存储,为了更好地讲解,请看下图:


笔试题(4)

int main()
{
    int a[3][2] = { (0, 1), (2, 3), (4, 5) };
    int* p;
    p = a[0];
    printf("%d", p[0]);
    return 0;
}
//程序输出的结果是什么?

答案:1

这里有一个陷阱,我们知道二维数组的定义方法应为int a[3][2]={{0,1},{2,3},{4,5}};

但是本题为int a[3][2] = { (0, 1), (2, 3), (4, 5) };

他实际上为int a[3][2] = { 1, 3, 5 };

即:

所以值为1。


笔试题(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;
}
//程序的结果是什么?

答案:FFFFFFFC,-4

根据前面学过的知识,我们知道p是数组指针,指向的元素类型为int [4]。

我们将p[4][2]也理解为二维数组,那么&p[4][2]就是指向p数组第5行第三列的元素,即为图中所示。

指针相减得到两个指针之间的元素个数,又因为地址是由低到高变化的,所以小地址减去大地址得到负数,为-4。

%p是打印地址的操作符,又因为-4在内存中存储的是补码:

1111 1111 1111 1111 1111 1111 1111 1100  //-4的补码

  F      F      F      F      F      F      F      C  

转化为16进制数字0xFFFFFFFC


笔试题(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

ptr1:

&aa取出的是整个数组的地址,+1指向元素'10'后面的元素,并强制转化为int*类型,打印时-1指向元素'10',解引用得到元素'10',所以值为10。

ptr2:

aa+1指向二维数组的第二行的一维数组,解引用得到该一维数组首元素的地址,也可以将*(aa+1)理解为aa[1],即为6的地址,打印时-1指向元素'5',解引用得到元素'5',所以值为5。

解引用一个数组(非数组名)的地址,得到的是该数组首元素的地址。


笔试题(7)

int main()
{
     char *a[] = {"work","at","alibaba"};
     char**pa = a;
     pa++;
     printf("%s\n", *pa);
     return 0;
}
//程序的结果是什么?

答案:at

a本质上是指针数组,该数组存放着三个字符串首个字符的地址。

pa指向的是数组a的首元素的地址,所以pa是一个二级指针,pa指向指向字符'w'的指针。

即:

所以得到的值为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

与第七题同理,难度略高。

如图,当我们知道指针之间的关系后,那么还需要攻克的就是符号优先级的问题。

1、**++cpp,++的优先级高于*,所以cpp先+1在连续两次解引用,得到元素'P'的地址,打印得到POINT。

2、*--*++cpp+3,cpp先++,再解引用,再--,再解引用,最后+3,得到元素'E'的地址,打印得到ER。

3、*cpp[-2]+3,可以理解为**(cpp-2)+3,得到元素'S'的地址,打印得到ST。

4、cpp[-1][-1]+1,可以理解为*(*(cpp-1)-1) +1,得到元素'E'的地址,打印得到EW。


小结:进阶指针的学习到这里就全部结束了,想要学习关于进阶指针的全部知识,欢迎到我的主页查询进阶指针(一)与进阶指针(二),接下来博主会持续更新C语言的更多知识, 关注博主不迷路🔥🔥🔥

目录
相关文章
|
17天前
使用指针访问数组元素
【10月更文挑战第30天】使用指针访问数组元素。
30 3
|
29天前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
48 0
|
4天前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
28 4
|
16天前
使用指针访问数组元素
【10月更文挑战第31天】使用指针访问数组元素。
29 2
|
25天前
|
算法 索引
单链表题+数组题(快慢指针和左右指针)
单链表题+数组题(快慢指针和左右指针)
29 1
|
30天前
|
存储 编译器 C语言
【c语言】数组
本文介绍了数组的基本概念及一维和二维数组的创建、初始化、使用方法及其在内存中的存储形式。一维数组通过下标访问元素,支持初始化和动态输入输出。二维数组则通过行和列的下标访问元素,同样支持初始化和动态输入输出。此外,还简要介绍了C99标准中的变长数组,允许在运行时根据变量创建数组,但不能初始化。
37 6
|
28天前
|
C语言
【c语言】指针就该这么学(3)
本文介绍了C语言中的函数指针、typedef关键字及函数指针数组的概念与应用。首先讲解了函数指针的创建与使用,接着通过typedef简化复杂类型定义,最后探讨了函数指针数组及其在转移表中的应用,通过实例展示了如何利用这些特性实现更简洁高效的代码。
17 2
|
28天前
|
C语言
如何避免 C 语言中的野指针问题?
在C语言中,野指针是指向未知内存地址的指针,可能引发程序崩溃或数据损坏。避免野指针的方法包括:初始化指针为NULL、使用完毕后将指针置为NULL、检查指针是否为空以及合理管理动态分配的内存。
|
28天前
|
C语言
C语言:哪些情况下会出现野指针
C语言中,野指针是指指向未知地址的指针,通常由以下情况产生:1) 指针被声明但未初始化;2) 指针指向的内存已被释放或重新分配;3) 指针指向局部变量,而该变量已超出作用域。使用野指针可能导致程序崩溃或不可预测的行为。
|
1月前
|
存储
如何使用指针数组来实现动态二维数组
指针数组可以用来实现动态二维数组。首先,定义一个指向指针的指针变量,并使用 `malloc` 为它分配内存,然后为每个子数组分配内存。通过这种方式,可以灵活地创建和管理不同大小的二维数组。