18. 《C语言》——【Nice2016年校招笔试题引发的思考】

2024-06-13 13:12

本文主要是介绍18. 《C语言》——【Nice2016年校招笔试题引发的思考】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


在这里插入图片描述


亲爱的读者,大家好!我是一名正在学习编程的高校生。在这个博客里,我将和大家一起探讨编程技巧、分享实用工具,并交流学习心得。希望通过我的博客,你能学到有用的知识,提高自己的技能,成为一名优秀的程序员。如果你有任何疑问或建议,请随时在评论区留言,让我们一起成长进步!现在,让我们开始这场知识之旅吧!


在这里插入图片描述

🏞️个人主页: FEN03
📊收入专栏: C语言

🌻🌼🌷🪻🌹🌷🌼🌻🌺🌹🪻🌼🌻🌺🌷🪻🌹🌷🌼🌻🌺🌻🌷🪻🌹🪻🌷🌼🌻🌻🌺🌼🌷🪻🌹🪻🌷


🎄文章目录

  • 🦊 前言 🍖
  • 🫎 Nice2016年校招笔试题 🥩
  • 🦦相同类型题目🥓
  • 👋🏻结束语🌹


🦊 前言 🍖

今天,有位朋友给我看了这道题,发现与Nice在2016年校招笔试题中的一道题是类似的,那么就让我们一起来探讨吧~ ⬇️⬇️⬇️


🫎 Nice2016年校招笔试题 🥩


下边代码为Nice在2016年校招笔试题中的其中一道,让我们来一起看看吧😃😃

要求是在LInux X86_64 gcc环境下,下面的程序会出现什么问题?运行结果是什么?为什么?

#include<stdio.h>
int main(int argc, char* argv[])
{long i;long a[16];for (i = 0; i <= 17; i++){a[i] = 0;printf("%d ", i);   }return 0;
}

当我看到这一道题目时,我就想起来有一道与它相似的,这道题目最早是出自于《C语言陷阱与缺陷》,那么接下来就让我们一起看看吧~ 🌹


🦦相同类型题目🥓


那么我在VS2022中,X86,Debug的环境下,并且编译器不做任何的优化,下面的代码会执行什么样的结果呢❓❓❓

#include <stdio.h>
int main()
{int i = 0;int arr[10] = {1,2,3,4,5,6,7,8,9,10};for (i = 0; i <= 12; i++){arr[i] = 0;printf("Nice\n");}return 0;
}

运行结果…

在这里插入图片描述

我们看到,程序运行结果,是死循环了,那么为什么会这样呢?

让我们调式观察吧⬇️⬇️⬇️

  1. 我们可以看到已经调式起来,并且可以在监视窗口看到i和arr的值
    在这里插入图片描述
  2. 下一步,到了for循环,for遍历数组,把数组的全部下标都改为了0,那么这时候数组还没有越界访问, 但是,for循环并没有到此结束,当我们继续会发生什么呢?
    在这里插入图片描述
  3. 当我继续时,这时候已经是越界访问了,并且把arr[10],arr[11],也改成了0,但是这时的for循环还没有结束,那么我再继续会发生什么❓❓❓
    在这里插入图片描述
  4. 那么我再继续,也就是最后一次for循环了
    在这里插入图片描述
    看下面动图,我进行最后一次for循环,arr[12]也是改为了0,突然 i 变量变为了0,并且与arr[12]的值是一样的了,这是为什么❓难道它们的地址是一样的吗❓
    在这里插入图片描述
  5. 当我把arr[12]与 i 的地址打开看看,可以发现它们的地址是一模一样的。
    在这里插入图片描述
    为了更为详细的解释,我们通过画图的方式进行讲解⬇️⬇️⬇️

分析:

  1. 首先整型变量i和整型数组arr,它们都是局部变量,那么局部变量是存放在栈区的,而栈区上内存的使用习惯是先使用高地址的空间再使用低地址的空间
  2. 那我们知道,栈区是存放局部变量和形式参数;堆区是存放动态内存开辟,如malloc,calloc,realloc,free;静态区是存放全局变量和由static修饰的静态变量。
  3. 我们可以把上面代码中它们在栈区的内存布局画出来
    我们通过该图,可以看到代码在栈区的内存布局,前边我们解释了,栈区是先使用高地址再使用低地址,我们按照创建变量的先后,分别把它们在内存布局画了出来,数组的地址是连续存放的并且是由低地址到高地址,
    那么通过不断的遍历数组,当arr[9]改为0时,这时候并没有越界访问,但是遍历还没结束,再次进行,arr[10],arr[11]都被改为了0,这已经是越界访问了,当再一次越界访问,看下图,这时arr[12]访问到了i 。所以,i 与 arr[12] 的地址是相同,既然地址相同,那么它们的值都是一样的,所以当i不断的++,arr[12]的值也跟着++,当arr[12] = 0 时,i也变为了0,最终不断的死循环。那么大家可能会有所疑惑,有这么巧吗❓
    在这里插入图片描述
  4. 我们看到 i 和 arr 中间空了2个int,其实这并不是巧合,至于它们中间空多大,取决于编译器的实现,在vc6.0中,相同的代码,i 和 arr 之间没有空隙;在gcc 中,i 和 arr 之间空1个int;在vs中空2个int 。

👋🏻结束语🌹


非常感谢您花时间阅读我的博客,希望我的分享能为您带来收获。如果您对本文有任何想法或疑问,欢迎在评论区留言交流。如果您喜欢我的博客,请继续关注,我会定期更新更多精彩内容。最后,别忘了行动起来的力量,让我们一起实践这些方法,见证自己的成长和进步!

再见,祝您生活愉快!


这篇关于18. 《C语言》——【Nice2016年校招笔试题引发的思考】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1057401

相关文章

GO语言短变量声明的实现示例

《GO语言短变量声明的实现示例》在Go语言中,短变量声明是一种简洁的变量声明方式,使用:=运算符,可以自动推断变量类型,下面就来具体介绍一下如何使用,感兴趣的可以了解一下... 目录基本语法功能特点与var的区别适用场景注意事项基本语法variableName := value功能特点1、自动类型推

GO语言中函数命名返回值的使用

《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

Go语言连接MySQL数据库执行基本的增删改查

《Go语言连接MySQL数据库执行基本的增删改查》在后端开发中,MySQL是最常用的关系型数据库之一,本文主要为大家详细介绍了如何使用Go连接MySQL数据库并执行基本的增删改查吧... 目录Go语言连接mysql数据库准备工作安装 MySQL 驱动代码实现运行结果注意事项Go语言执行基本的增删改查准备工作

Go语言使用Gin处理路由参数和查询参数

《Go语言使用Gin处理路由参数和查询参数》在WebAPI开发中,处理路由参数(PathParameter)和查询参数(QueryParameter)是非常常见的需求,下面我们就来看看Go语言... 目录一、路由参数 vs 查询参数二、Gin 获取路由参数和查询参数三、示例代码四、运行与测试1. 测试编程路

Go语言使用net/http构建一个RESTful API的示例代码

《Go语言使用net/http构建一个RESTfulAPI的示例代码》Go的标准库net/http提供了构建Web服务所需的强大功能,虽然众多第三方框架(如Gin、Echo)已经封装了很多功能,但... 目录引言一、什么是 RESTful API?二、实战目标:用户信息管理 API三、代码实现1. 用户数据

Go语言网络故障诊断与调试技巧

《Go语言网络故障诊断与调试技巧》在分布式系统和微服务架构的浪潮中,网络编程成为系统性能和可靠性的核心支柱,从高并发的API服务到实时通信应用,网络的稳定性直接影响用户体验,本文面向熟悉Go基本语法和... 目录1. 引言2. Go 语言网络编程的优势与特色2.1 简洁高效的标准库2.2 强大的并发模型2.

Go语言使用sync.Mutex实现资源加锁

《Go语言使用sync.Mutex实现资源加锁》数据共享是一把双刃剑,Go语言为我们提供了sync.Mutex,一种最基础也是最常用的加锁方式,用于保证在任意时刻只有一个goroutine能访问共享... 目录一、什么是 Mutex二、为什么需要加锁三、实战案例:并发安全的计数器1. 未加锁示例(存在竞态)

C语言自定义类型之联合和枚举解读

《C语言自定义类型之联合和枚举解读》联合体共享内存,大小由最大成员决定,遵循对齐规则;枚举类型列举可能值,提升可读性和类型安全性,两者在C语言中用于优化内存和程序效率... 目录一、联合体1.1 联合体类型的声明1.2 联合体的特点1.2.1 特点11.2.2 特点21.2.3 特点31.3 联合体的大小1

Go语言使用select监听多个channel的示例详解

《Go语言使用select监听多个channel的示例详解》本文将聚焦Go并发中的一个强力工具,select,这篇文章将通过实际案例学习如何优雅地监听多个Channel,实现多任务处理、超时控制和非阻... 目录一、前言:为什么要使用select二、实战目标三、案例代码:监听两个任务结果和超时四、运行示例五

C语言中%zu的用法解读

《C语言中%zu的用法解读》size_t是无符号整数类型,用于表示对象大小或内存操作结果,%zu是C99标准中专为size_t设计的printf占位符,避免因类型不匹配导致错误,使用%u或%d可能引发... 目录size_t 类型与 %zu 占位符%zu 的用途替代占位符的风险兼容性说明其他相关占位符验证示