热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

全局变量与常量在内存中的布局分析及应用

本文详细探讨了全局变量与常量在内存中的存储布局及其应用。通过分析不同编译器和操作系统对全局变量与常量的处理方式,揭示了它们在内存中的具体分配机制。此外,文章还讨论了这些布局对程序性能和安全的影响,并提供了优化建议,帮助开发者更好地理解和利用全局变量与常量的内存管理。
094bc4e2c42261eb2d74715cac8e8d19.gif

元旦快乐

happy new year

a36c99fbfb342cf66aa6847363b01682.png

1bdf2ff8c0474b68dff2fb4792d5b7be.png687f8dfb623ba4a788605d521a08adb1.png

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

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

997fd2ff7a2ad1b506c20a7f4f824894.png68595bb0a742d3dc6ad3ad7f3409f889.png094bc4e2c42261eb2d74715cac8e8d19.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




推荐阅读
author-avatar
手机用户2602899031
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有