全局常量_全局变量和常量在内存中的分布

本文探讨了全局变量和常量在内存中的分布情况。内容包括内存的四个主要区域:代码段、数据段(已初始化和未初始化)、栈和堆。重点讲解了数据段中的全局变量和常量,以及它们的读写、执行和共享属性。通过实例展示了不同类型的全局变量和常量在内存中如何按其属性连续分配。此外,还提到了常量在内存中的独立位置。
摘要由CSDN通过智能技术生成
094bc4e2c42261eb2d74715cac8e8d19.gif

元旦快乐

happy new year

a36c99fbfb342cf66aa6847363b01682.png

1bdf2ff8c0474b68dff2fb4792d5b7be.png 687f8dfb623ba4a788605d521a08adb1.png

新年最大的心愿就是一家人平平安安,幸福开心。

大人物小城梦公众号,祝福大家,元旦快乐。长大后,因为各种原因,陪伴家人很少,在以后的每年,一定多拿出一些时间陪陪家人。

997fd2ff7a2ad1b506c20a7f4f824894.png 68595bb0a742d3dc6ad3ad7f3409f889.png 094bc4e2c42261eb2d74715cac8e8d19.gif

内存结构

memory  structure

    内存中,有代码段,数据段,栈,堆。数据段分为已初始化和未初始化。

    内存中属性很多,但作为程序员,只重点关注读(read),写(write),执行(execyte),共享(share),这四个属性。英语取开头字母,缩写为rwes。

    代码段放程序的代码,属性是可读可执行。

    数据段放全局数据,静态数据和常量。未初始化部分,属性是可读写。已初始化部分,分两部分,一部分属性是可读写,一部分属性是只读。

    栈放参数,局部变量,保存的环境和返回地址,属性是可读写。

    堆的属性是可读写。

    这里的属性是初始状态,在程序运行过程中,操作系统有一定的机制可以调整属性。

7f7d859d04ad7d702a1df5c9ee00a4c4.png

094bc4e2c42261eb2d74715cac8e8d19.gif

全局变量和常量的内存分布

Memory distribution for global variables and constants

    游离在其他函数以外,定义的变量,叫做全局变量。全局变量可以被所有函数同时访问。

    编写程序:

#include #include int g_nTest1 = 0x53801204;int g_nTest2 = 0x41201314;void foo(){  printf("%p:", &g_nTest1);  printf("%d\r\n", g_nTest1);}int main(){  int nTest = 999;  printf("%p:", &nTest);  printf("%d\r\n", nTest);  printf("%p:", &g_nTest1);  printf("%d\r\n", g_nTest1);  foo();  system("pause");  return 0;}

    按 F10 进行单步调试,可以看到 nTest 的地址是 0019ff2c

fb848f6554fe74ddd8f44f2b9fe876d9.png

    g_nTest1 的地址是 00429098 ,g_nTest2 的地址是 0042909c。两个是相邻的。

    由此可见,编译器分配变量的原则:按同内存属性类型分配

d5e009d4c37ba61db4ec7b8f1d907d4b.png

    在原有代码中,定义一个字符串 Hello

#include #include int g_nTest1 = 0x53801204;char g_szHello[] = "Hello";int g_nTest2 = 0x41201314;void foo(){  printf("%p:", &g_nTest1);  printf("%d\r\n", g_nTest1);}int main(){  int nTest = 999;  printf("%p:", &nTest);  printf("%d\r\n", nTest);  printf("%p:", &g_nTest1);  printf("%d\r\n", g_nTest1);  foo();  system("pause");  return 0;}

    按 F10 进行单步调试,从内存结构可以看出,虽然数据类型不一致,有 int 类型,有字符串。但是它们都是已初始化的全局数据,所以分配的空间是连续的。

be40e0a94c0bd00b190f82568209d314.png

    将代码中的 g_nTest2 的初值去掉

#include #include int g_nTest1 = 0x53801204;char g_szHello[] = "Hello";int g_nTest2;void foo(){  printf("%p:", &g_nTest1);  printf("%d\r\n", g_nTest1);}int main(){  int nTest = 999;  printf("%p:", &nTest);  printf("%d\r\n", nTest);  printf("%p:", &g_nTest1);  printf("%d\r\n", g_nTest1);  foo();  system("pause");  return 0;}

    因为 g_nTest1 和 g_szHello 是已初始化全局数据,而 g_nTest2 是未初始化全局数据,所以 g_nTest1 和 g_szHello 连续放在已初始化全局数据区,g_nTest2 独自放在未初始化全局数据区。

    按 F10 进行单步调试,g_nTest1 的地址是 00426a30 。g_szHello 的地址是 00426a34,和 g_nTest1 是连续排列的。而 g_nTest2 的地址却是 00429e60。

09467e14768e053e2adf64d71a5b64a5.png

8f3a06752900d4f8f5f203f2017156ee.png

    添加常量 g_nTest = 0x13145201 在代码中

#include #include int g_nTest1 = 0x53801204;char g_szHello[] = "Hello";int g_nTest2;const int g_nTest = 0x13145201void foo(){  printf("%p:", &g_nTest1);  printf("%d\r\n", g_nTest1);}int main(){  int nTest = 999;  printf("%p:", &nTest);  printf("%d\r\n", nTest);  printf("%p:", &g_nTest1);  printf("%d\r\n", g_nTest1);  foo();  system("pause");  return 0;}

    按 F10 进行单步调试,可见,初始化全局数据在一个内存空间,未初始化全局数据在一个内存空间,常量在一个内存中间。

91e27daf6b3a981c7adc044da1e73e97.png

39bdcb94bac3bc33be1532060e718ec5.png                          

ca3277f6777ccd4009794e935afdb2b0.png

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值