zoukankan      html  css  js  c++  java
  • C语言的未初始化的数组的值为什么是随机的

    突然想起来前几天同学问我为什么没有初始化的数组的值是随机的,发现这个困惑自己也是存在的,所以自己总结的心得.

     1. 首先,并不是所有未初始化的数组的值都是随机的.对于没有初始化的数组,分两种情况:

        (1)全局数组,也就是定义在main函数外面的数组,元素的默认值是全部为0的

        (2)局部数组,定义在函数内部的数组,其值默认是随机的. 

    #include <stdio.h>
    
    #define LENGTH 5
    int a[LENGTH];
    
    int main()
    {
        for(int i=0;i<LENGTH;i++){
            printf("%d ",a[i]);
        }
        printf("
    ");
    
        int b[LENGTH];
        for(int i=0;i<LENGTH;i++){
            printf("%d ",b[i]);
        }
    }
    0 0 0 0 0
    4200814 4200720 49 8 41
    Process returned 0 (0x0)   execution time : 0.739 s
    Press any key to continue.

      从代码结果可以清楚地看出来,全局数组与局部数组的默认值是不同的.

    2.我们接下来再说局部数组为什么是随机的.局部数组是放在栈区的,而全局数组是放在静态区的.

      因为局部数组放在栈区,栈的操作就是入栈和出栈.我们声明数组,其实只是移动栈顶指针.而栈内的数据是上一次出栈时候遗留的数据.栈不会清空.所以数据是随机的.下面用一段代码说明.

    #include <stdio.h>
    void test();
    int main()
    {
        test();
        test();
    }
    
    void test()
    {
        int a[5];
        for(int i=0;i<5;i++){
            printf("%d ",a[i]);
        }
        printf("
    ");
        for(int i=0;i<5;i++){
            a[i] =i;
        }
        for(int i=0;i<5;i++){
            printf("%d ",a[i]);
        }
        printf("
    ");
    }
    //输出结果
    1944480698 1944480941 4200720 6356884 4200814
    0 1 2 3 4
    0 1 2 3 4
    0 1 2 3 4

      当我们连续两次调用一个函数的时候.我们发现只有第一次的值似乎是随机的.因为我们两次相同的操作对栈的地址操作也是相同的.我们第一次函数对栈的修改并没有被栈清空,所以第二次的随机值就是第一次最后的数据了.

    3.第二次更新,今天又想到了一个骚操作来证明.用的是c++.但是道理都是想通的.

    因为重复运行这个代码局部数组的地址总是不变的,我的是0x6afed0. 可以通过

    printf("%x",a);   //a为数组名

    来获得数组名,用c++的地址转换将数组首地址赋值给指针sp,然后通过sp操作改变数组第一位的值.以此来解释未初始化的数组的值随机的来源.我们从最终结果可以看出来.内存空间中这个地址当时的值是什么.因为未初始化,所以显示的值就是什么.

    #include <stdio.h>
    #include <iostream>
    
    void test();
    int main()
    {
        int* sp=reinterpret_cast<int *>(0x6afed0);
        *sp = 168;
        test();
        *sp = 113;
        test();
    }
    
    void test()
    {
        int a[5];
        for(int i=0;i<5;i++){
            printf("%d ",a[i]);
        }
        printf("
    ");
        for(int i=0;i<5;i++){
            a[i] =i;
        }
        for(int i=0;i<5;i++){
            printf("%d ",a[i]);
        }
        printf("
    ");
    }
    
    //输出结果
    168 7012032 7012088 7012300 1981401632
    0 1 2 3 4
    113 1 2 3 4
    0 1 2 3 4
  • 相关阅读:
    xsd的解释说明
    SDUT 2498-AOE网上的关键路径(spfa+字典序路径)
    java实现各种数据统计图(柱形图,饼图,折线图)
    软件測试计划模板
    范式图形辨析
    Android做法说明(3)---Fragment使用app袋或v4包解析
    登录同步多个副本如何实现的拷贝数发生变化分布式
    ZOJ1463:Brackets Sequence(间隙DP)
    jquery php 百度搜索框智能提示效果
    Hibernate在关于一对多,多对一双向关联映射
  • 原文地址:https://www.cnblogs.com/Triomphe/p/9374539.html
Copyright © 2011-2022 走看看