指针和数组笔试题解析【下】

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

1.数组笔试题

之前,我们已经了解了数组名的意义,这里不再继续阐述。

//二维数组
//在内存中也是连续存放,二维数组是一维数组的数组
int a[3][4] = { 0 };
printf("%zd\n", sizeof(a));
//a表示整个数组,计算的是整个数组的大小,单位是字节---48字节==12个元素*每个元素的大小4字节
printf("%zd\n", sizeof(a[0][0]));
//表示数组的第一行第一个元素的大小,4个字节
printf("%zd\n", sizeof(a[0]));
//二维数组的数组名是a,第一行的数组名是a[0],第二行的数组名是a[1],第三行的数组名是a[2]
//所以a[0]作为第一行的数组名,单独放在sizeof内部,计算的就是整个第一行的大小
//所以是16个字节
printf("%zd\n", sizeof(a[0] + 1));
//a[0]并非单独放在sizeof内部,也没有&,所以a[0]表示首元素的地址,也就是第一行第一个元素的地址
//a[0]===&a[0][0]
//a[0]+1==&a[0][1]
//是地址就是4/8个字节
printf("%zd\n", sizeof(*(a[0] + 1)));
//a[0]+1是第一行第二个元素的地址,解引用就是第一行第二个元素,大小就是4个字节
printf("%zd\n", sizeof(a + 1));
//a作为二维数组的数组名,并没有单独放在sizeof内部,也没有&,表示的是数组首元素的地址(第一行的地址),+1就是第二行的地址,是地址就是4/8个字节,a的类型是 int(*)[4]
printf("%zd\n", sizeof(*(a + 1)));
//a+1是二维数组第二行的地址,解引用计算的是第二行数组的大小,就是16个字节
printf("%zd\n", sizeof(&a[0] + 1));
//a[0]是第一行的数组名,取出的是第一行的地址,+1就是第二行的地址,是地址就是4/8个字节
printf("%zd\n", sizeof(*(&a[0] + 1)));
//计算的是第二行整个数组的地址,解引用访问的是第二行,计算的大小就是16个字节
printf("%zd\n", sizeof(*a));
//a表示数组首元素的地址,就是第一行的地址,*a就是第一行,解引用计算的就是第一行的大小,就是16个字节
printf("%zd\n", sizeof(a[3]));
//不会越界,sizeof不会真的去访问大小,是根据类型去计算的大小,所以大小是16字节

2.指针笔试题

2.1

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

image.png

所以结果是 2     5.。


2.2

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

printf("%p\n", p + 0x1);

                  结构体指针+1,跳过一个结构体的大小(20字节),所以p+0x1结果就是0x100014。


printf("%p\n", (unsigned long)p + 0x1);

                 把p强制转换成无符号整型,把指针强制转换成整型,+0x1结果就是0x100001。


printf("%p\n", (unsigned int*)p + 0x1);

                 把p强制转换成无符号整型指针,+1跳过4个字节,所以结果就是0x100004。



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

int *ptr1 = (int *)(&a + 1);

image.png


                                                       int *ptr2 = (int *)((int)a + 1);


a是首元素地址,被强制转换成int,假设a的地址是0x0012FF40,+1就是0x0012FF41,然后又被强制转换成int*指针赋值给ptr2,此时ptr2指向的就是内存中存放1的第二个字节,见下图。


image.png


image.png

目录
相关文章
|
2月前
使用指针访问数组元素
【10月更文挑战第30天】使用指针访问数组元素。
46 3
|
2月前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
2月前
|
存储 C语言 计算机视觉
在C语言中指针数组和数组指针在动态内存分配中的应用
在C语言中,指针数组和数组指针均可用于动态内存分配。指针数组是数组的每个元素都是指针,可用于指向多个动态分配的内存块;数组指针则指向一个数组,可动态分配和管理大型数据结构。两者结合使用,灵活高效地管理内存。
|
2月前
|
容器
在使用指针数组进行动态内存分配时,如何避免内存泄漏
在使用指针数组进行动态内存分配时,避免内存泄漏的关键在于确保每个分配的内存块都能被正确释放。具体做法包括:1. 分配后立即检查是否成功;2. 使用完成后及时释放内存;3. 避免重复释放同一内存地址;4. 尽量使用智能指针或容器类管理内存。
|
2月前
|
存储 NoSQL 编译器
C 语言中指针数组与数组指针的辨析与应用
在C语言中,指针数组和数组指针是两个容易混淆但用途不同的概念。指针数组是一个数组,其元素是指针类型;而数组指针是指向数组的指针。两者在声明、使用及内存布局上各有特点,正确理解它们有助于更高效地编程。
|
1月前
|
存储 程序员 C++
深入解析C++中的函数指针与`typedef`的妙用
本文深入解析了C++中的函数指针及其与`typedef`的结合使用。通过图示和代码示例,详细介绍了函数指针的基本概念、声明和使用方法,并展示了如何利用`typedef`简化复杂的函数指针声明,提升代码的可读性和可维护性。
83 0
|
2月前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
70 4
|
2月前
使用指针访问数组元素
【10月更文挑战第31天】使用指针访问数组元素。
54 2
|
2月前
|
算法 索引
单链表题+数组题(快慢指针和左右指针)
单链表题+数组题(快慢指针和左右指针)
43 1
|
3月前
|
人工智能 前端开发 JavaScript
拿下奇怪的前端报错(一):报错信息是一个看不懂的数字数组Buffer(475) [Uint8Array],让AI大模型帮忙解析
本文介绍了前端开发中遇到的奇怪报错问题,特别是当错误信息不明确时的处理方法。作者分享了自己通过还原代码、试错等方式解决问题的经验,并以一个Vue3+TypeScript项目的构建失败为例,详细解析了如何从错误信息中定位问题,最终通过解读错误信息中的ASCII码找到了具体的错误文件。文章强调了基础知识的重要性,并鼓励读者遇到类似问题时不要慌张,耐心分析。

热门文章

最新文章

推荐镜像

更多