C语言入坑指南-被遗忘的初始化

前言

什么是初始化?为什么要初始化?静态变量和局部变量的初始化又有什么区别?实际应用中应该怎么做?本文将一一回答这些问题。

什么是初始化

初始化指的是对数据对象或者变量赋予初始值。例如:

int value = 8; //声明整型变量并初始化为8int arr[] = {1,2,3}; //声明整型数组arr,并初始化其值为1,2,3

为什么要初始化

我们来看一个示例程序。
test0.c程序清单如下:

#include <stdio.h>#include <stdlib.h>int main(void){    int sum;    int randNum;    while(10 > sum)    {        randNum =  rand() % 10;

        sum += randNum;        printf("rand num is %d,sum is %d\n",randNum,sum);    }    printf("the final sum is %d\n",sum);    return 0;}

程序随机产生0到9的数字,使得sum的值大于或等于10时,退出程序。
编译并运行:

gcc  -o test0 test0.c./test0

运行结果如下(每次运行结果可能不同):

rand num is 3,sum is -4040865rand num is 6,sum is -4040859rand num is 7,sum is -4040852rand num is 5,sum is -4040847rand num is 3,sum is -4040844rand num is 5,sum is -4040839(省略其他内容)

从运行结果来看,程序并没有达到我们的预期,这是为什么呢?

很多读者可能已经知道,问题在于声明sum之后,没有为其赋初始值,在这样的情况下,sum的值是随机的,因此在一开始sum可能是一个很小的负数,导致多次循环出现。很显然,初始化避免使用了变量的“脏值”。而将sum的声明改成如下定义即可:

int sum = 0;

如果将sum声明为静态变量,情况又会如何呢?

//test1.c#include <stdio.h>#include <stdlib.h>int main(void){    static int sum;    int randNum;    while(10 > sum)    {        randNum =  rand() % 10;

        sum += randNum;        printf("rand num is %d,sum is %d\n",randNum,sum);    }    printf("the final sum is %d\n",sum);    return 0;}

编译并运行:

rand num is 3,sum is 3rand num is 6,sum is 9rand num is 7,sum is 16the final sum is 16

在这种情况下,程序是能够符合我们预期的结果,这又是为什么呢?原因在于静态变量会被默认初始化。例如,int类型会被初始化为0。那么问题来了:

  • 为什么局部变量未初始化的时候的值是“脏值”?
  • 静态变量和局部变量为什么又不一样呢?

在解答上面这两个问题之前,我们需要简单了解一下程序的存储空间布局。

程序的存储空间布局

C程序主要由以下几部分组成:

  • 正文段。即机器指令部分,为防止意外被修改,设为只读。
  • 初始化数据段。它包含了程序中需要明确赋初值的静态变量。
  • 未初始化数据段。它包含了程序中未赋初值的或初始化为0的静态变量,在程序开始执行之前,内核将此段中的数据初始化为0。
  • 栈。它保存了自动(局部)变量以及函数调用所要的信息。
  • 堆。用于动态内存分配。例如使用malloc函数进行内存分配。

其中,正文段和数据段的内容是“静态”的,因为在程序被编译出来之后,在整个程序地址就确定了,而堆栈中的内容是”动态”变化的,它随着进行的运行而不断变化着,再加上栈随机化的策略,使得程序每次运行时,栈的地址也是不确定的。

局部变量和静态变量的初始化有何不同

有了前面的铺垫,就很好理解两者的差别了。
未初始化的局部变量位于栈中,它的位置是不确定的,因此其值也是不确定的。当然,在windows下它的值是0xcccccccc,而“烫”字在MBCS字符集中的值为0xcccccccc,你说巧不巧?

而静态变量就不一样的,它的地址是确定的,并且存放在了数据段,而程序在运行之前,未初始化数据段的内容可以很方便地统一被初始化为0。这也就解释了前面的两个示例程序的结果为什么会不一样。我们加上一些打印,来看一看是否真的如此?

//test2.c#include <stdio.h>#include <stdlib.h>int main(void){    static int sum;    int randNum;    while(10 > sum)    {        randNum =  rand() % 10;

        sum += randNum;        printf("rand num is %d,sum is %d\n",randNum,sum);    }    printf("the final sum is %d\n",sum);    printf("sum addr %p,randNum addr %p\n",&sum,&randNum);    return 0;}

编译并运行:

gcc -o test2 test2.c

运行结果1:

rand num is 3,sum is 3rand num is 6,sum is 9rand num is 7,sum is 16the final sum is 16sum addr 0x60104c,randNum addr 0x7ffd0ea8cf54

运行结果2:

rand num is 3,sum is 3rand num is 6,sum is 9rand num is 7,sum is 16the final sum is 16sum addr 0x60104c,randNum addr 0x7ffff5e3ddb4

在这里,sum是静态局部变量,而randNun是局部变量(自动变量),因此可以发现,sum的地址值总是不变的,而randNum的值却不断变化着。我们也可以通过nm命令查看sum的地址:

nm test2 |grep sum000000000060104c b sum.2805

总结

我们来总结一下本文的主要内容:

  • 如果变量是静态的,它会被初始化为0;如果变量是自动的,它不会被初始化。
  • 静态的变量包括全局变量、静态全局变量、静态局部变量。
  • 使用局部变量之前对其进行初始化,避免使用“脏值”。
  • 从可读性考虑,静态变量也建议显示初始化。
  • 初始化为0的静态变量仍然存在未初始化数据段中(BSS段)。

送几句熟悉的话给大家:

手持两把锟斤拷,口中疾呼烫烫烫。脚踏千朵屯屯屯,笑看万物锘锘锘。

思考

test1.c的代码运行结果每次都一样吗?为什么?该如何修改才能使得每次的运行结果不一样?

栈随机化的作用是什么?

推荐阅读:

C语言入坑指南-数组之谜

一个命令帮你对文本排序

如何理解 Linux shell中“2>&1”?

推荐一款强大的在线编译器

Linux常用命令--系统状态篇

关注公众号【编程珠玑】,第一时间获取更多原创技术文章

原文地址:https://www.cnblogs.com/bianchengzhuji/p/9960183.html

时间: 2024-10-13 06:47:22

C语言入坑指南-被遗忘的初始化的相关文章

eclipse中导入外部包却无法查看对应源码或Javadoc的 入坑指南

eclipse中导入外部包却无法查看对应源码或Javadoc的 入坑指南 出现这个错误的原因是,你虽然导入了.jar包,但没有配置对应的Javadoc或源码路径,所以在编辑器中无法查看源 码和对应API.接下来我们一起解决这个问题... 在项目名称上右击→ 新建→ 文件夹→ 文件名写lib→ 点击完成 然后把你下载的jar包,复制黏贴到这个lib文件夹 右击lib中的源码包→ 构建路径→ 添加至构建路径,自动生成一个"引用的库" 右击 "引用的库" 中的jar包→

Rust入坑指南:亡羊补牢

如果你已经开始学习Rust,相信你已经体会过Rust编译器的强大.它可以帮助你避免程序中的大部分错误,但是编译器也不是万能的,如果程序写的不恰当,还是会发生错误,让程序崩溃.所以今天我们就来聊一聊Rust中如何处理程序错误,也就是所谓的"亡羊补牢". 基础概念 在编程中遇到的非正常情况通常可以分为三类:失败.错误.异常. Rust中用两种方式来消除失败:强大的类型系统和断言. 对于类型系统,熟悉Java的同学应该比较清楚.例如我们给一个接收参数为int的函数传入了字符串类型的变量.这是

Kotlin快速入坑指南(干货型文档)

<p style="text-align:center;color:#42A5F5;font-size:2em;font-weight: bold;">前言 即使每天10点下班,其实需求很多,我也要用这腐朽的声带喊出:我要学习,我要写文章!! 又是一篇Kotlin的文章,为啥...还不是因为工作需要.毫无疑问,最好的学习方式是通过官方文档去学习.不过个人觉得官方文档多多少少有一些不够高效. 中文官方文档 因此这篇是从我学习的个人视角以文档的形式去输出Kotlin语言基础的学

Rust入坑指南:坑主驾到

欢迎大家和我一起入坑Rust,以后我就是坑主,我主要负责在前面挖坑,各位可以在上面看,有手痒的也可以和我一起挖.这个坑到底有多深?我也不知道,我是抱着有多深就挖多深的心态来的,下面我先跳了,各位请随意. Rust简介 众所周知,在编程语言中,更易读的高级语言和控制底层资源的低级语言是一对矛盾体.Rust想要挑战这一现状,它尝试为开发者提供更好的体验的同时给予开发者控制底层细节的权限(比如内存使用). 低级语言在开发过程中很容易出现各种细微的错误,它们难以发现但是可能影响巨大.其他大部分低级语言只

猿说摄影(上)--入坑指南

最近师弟师妹们以及复读的童鞋临近毕业,有的想买相机拍拍毕业照,记录一下旅行毕业游之类的.五一放假,咱就先不聊技术,聊一下摄影,不过摄影也是一个技术活,而且烧钱.摄影穷三代,单反毁一生.相机贵吗?贵,但贵的不只是相机,还有镜头.为什么这么说呢?大家也知道,单反和微单都是可以更换镜头的.一旦入坑,除了买相机同时买的套头(标准变焦镜头)之外,你很可能会接下来陆陆续续地买其它镜头→_→想拍漂亮的人物,你需要大光圈的定焦镜头:想拍壮阔的风景,你需要广角镜头:想拍飞禽走兽,你需要长焦镜头:你可能还要拍点小花

Docker入坑指南之RUN

总有一些场景,我们需要自己制作一个镜像,可以快速还原环境,又不想被其他因素干扰镜像的纯净,这个时候,就可以选择Docker了,启动便捷,镜像还原很快捷,除了上手不容易. 最近入坑研究了一番,小有心得,故写一篇杂文,记录自己的踩坑经历. 安装Docker的过程可以参考其他前辈的文章,不再赘述,从实战角度说,如何构建一个自用的Docker镜像. 首选说一下Docker的几个名词,仓库是管理镜像的,容器是镜像启动后的,镜像就是最干净的环境,镜像启动之后变成容器. docker的run是启动镜像的介质,

NW.js 入坑指南

NW.js是什么? NW.js 是基于 Chromium 和 Node.js 运行的, 以前也叫nodeWebkit.这就给了你使用HTML和JavaScript来制作桌面应用的可能.在应用里你可以直接调用Node.js的各种api以及现有的第三方包.因为Chromium和 Node.js 的跨平台,那么你的应用也是可以跨平台的.现在已经有很多知名的应用是基于NW.js实现,这是官方统计的一些列表: https://github.com/nwjs/nw.js/wiki/List-of-apps-

Openstack入坑指南

什么是云计算 概念 云计算是一种基于互联网的计算方式,通过这种方式,共享的软硬件资源和信息,可以按需求提供给计算机和其他设备.用户不需要了解”云“中的基础设施细节,不必具有相应的专业知识,也无需直接控制.云计算描述了一种基于互联网的新的IT服务增加.使用和交付模式. 我们举一个例子来理解云计算,云计算中的”云“可以理解为天上的云,天上的云可以变成雨水降落到地上,落到地上的水蒸发后又变成云彩.这样就形成了一个循环. 这里的雨水表示计算资源,比如虚拟机.存储.网络等等. 云变水的过程表示获取资源的过

[Web 前端] React Router v4 入坑指南

cp from : https://www.jianshu.com/p/6a45e2dfc9d9 万恶的根源 距离React Router v4 正式发布也已经过去三个月了,这周把一个React的架子做了升级,之前的路由用的还是v2.7.0版的,所以决定把路由也升级下,正好“尝尝鲜”... 江湖传言,目前官方同时维护 2.x 和 4.x 两个版本.(ヾ(??﹏?)??咦,此刻相信机智如我的你也会发现,ReactRouter v3 去哪儿了?整丢了??巴拉出锅了???敢不敢给我个完美的解释!?)事