【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语言的更多知识, 关注博主不迷路🔥🔥🔥

目录
相关文章
|
5天前
|
存储 C语言
C语言32位或64位平台下指针的大小
在32位平台上,C语言中指针的大小通常为4字节;而在64位平台上,指针的大小通常为8字节。这反映了不同平台对内存地址空间的不同处理方式。
|
1天前
|
存储 编译器 C语言
【c语言】数组
本文介绍了数组的基本概念及一维和二维数组的创建、初始化、使用方法及其在内存中的存储形式。一维数组通过下标访问元素,支持初始化和动态输入输出。二维数组则通过行和列的下标访问元素,同样支持初始化和动态输入输出。此外,还简要介绍了C99标准中的变长数组,允许在运行时根据变量创建数组,但不能初始化。
17 5
|
4天前
|
存储
如何使用指针数组来实现动态二维数组
指针数组可以用来实现动态二维数组。首先,定义一个指向指针的指针变量,并使用 `malloc` 为它分配内存,然后为每个子数组分配内存。通过这种方式,可以灵活地创建和管理不同大小的二维数组。
|
4天前
|
存储
如何通过指针数组来实现二维数组?
介绍了二维数组和指针数组的概念及其区别,详细讲解了如何使用指针数组模拟二维数组,包括定义与分配内存、访问和赋值元素、以及正确释放内存的步骤,适用于需要动态处理二维数据的场景。
|
4天前
|
存储 算法 C语言
C语言:什么是指针数组,它有什么用
指针数组是C语言中一种特殊的数据结构,每个元素都是一个指针。它用于存储多个内存地址,方便对多个变量或数组进行操作,常用于字符串处理、动态内存分配等场景。
|
5天前
|
存储 C语言
C语言指针与指针变量的区别指针
指针是C语言中的重要概念,用于存储内存地址。指针变量是一种特殊的变量,用于存放其他变量的内存地址,通过指针可以间接访问和修改该变量的值。指针与指针变量的主要区别在于:指针是一个泛指的概念,而指针变量是具体的实现形式。
|
5天前
|
C语言
C语言指针(3)
C语言指针(3)
9 1
|
5天前
|
C语言
C语言指针(2)
C语言指针(2)
9 1
|
8天前
|
存储 C语言
C语言:一维数组的不初始化、部分初始化、完全初始化的不同点
C语言中一维数组的初始化有三种情况:不初始化时,数组元素的值是随机的;部分初始化时,未指定的元素会被自动赋值为0;完全初始化时,所有元素都被赋予了初始值。
|
10天前
魔法指针 之 二级指针 指针数组
魔法指针 之 二级指针 指针数组
13 1