热门标签 | 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




推荐阅读
  • Linux环境下进程间通信:深入解析信号机制
    本文详细探讨了Linux系统中信号的生命周期,从信号生成到处理函数执行完毕的全过程,并介绍了信号编程中的注意事项和常见应用实例。通过分析信号在进程中的注册、注销及处理过程,帮助读者理解如何高效利用信号进行进程间通信。 ... [详细]
  • 本文介绍如何从字符串中移除大写、小写、特殊、数字和非数字字符,并提供了多种编程语言的实现示例。 ... [详细]
  • 本题来自WC2014,题目编号为BZOJ3435、洛谷P3920和UOJ55。该问题描述了一棵不断生长的带权树及其节点上小精灵之间的友谊关系,要求实时计算每次新增节点后树上所有可能的朋友对数。 ... [详细]
  • Linux环境下C语言实现定时向文件写入当前时间
    本文介绍如何在Linux系统中使用C语言编程,实现在每秒钟向指定文件中写入当前时间戳。通过此示例,读者可以了解基本的文件操作、时间处理以及循环控制。 ... [详细]
  • 丽江客栈选择问题
    本文介绍了一道经典的算法题,题目涉及在丽江河边的n家特色客栈中选择住宿方案。两位游客希望住在色调相同的两家客栈,并在晚上选择一家最低消费不超过p元的咖啡店小聚。我们将详细探讨如何计算满足条件的住宿方案总数。 ... [详细]
  • 在进行QT交叉编译时,可能会遇到与目标架构不匹配的宏定义问题。例如,当为ARM或MIPS架构编译时,需要确保使用正确的宏(如QT_ARCH_ARM或QT_ARCH_MIPS),而不是默认的QT_ARCH_I386。本文将详细介绍如何正确配置编译环境以避免此类错误。 ... [详细]
  • JSOI2010 蔬菜庆典:树结构中的无限大权值问题
    本文探讨了 JSOI2010 的蔬菜庆典问题,主要关注如何处理非根非叶子节点的无限大权值情况。通过分析根节点及其子树的特性,提出了有效的解决方案,并详细解释了算法的实现过程。 ... [详细]
  • 本题要求实现一个函数,用于检查给定的字符串是否为回文。回文是指正向和反向读取都相同的字符串。例如,“XYZYX”和“xyzzyx”都是回文。 ... [详细]
  • 深入解析Java枚举及其高级特性
    本文详细介绍了Java枚举的概念、语法、使用规则和应用场景,并探讨了其在实际编程中的高级应用。所有相关内容已收录于GitHub仓库[JavaLearningmanual](https://github.com/Ziphtracks/JavaLearningmanual),欢迎Star并持续关注。 ... [详细]
  • 本文探讨了如何通过预处理器开关选择不同的类实现,并解决在特定情况下遇到的链接器错误。 ... [详细]
  • 本文介绍如何利用栈数据结构在C++中判断字符串中的括号是否匹配。通过顺序栈和链栈两种方式实现,并详细解释了算法的核心思想和具体实现步骤。 ... [详细]
  • 在高并发需求的C++项目中,我们最初选择了JsonCpp进行JSON解析和序列化。然而,在处理大数据量时,JsonCpp频繁抛出异常,尤其是在多线程环境下问题更为突出。通过分析发现,旧版本的JsonCpp存在多线程安全性和性能瓶颈。经过评估,我们最终选择了RapidJSON作为替代方案,并实现了显著的性能提升。 ... [详细]
  • 本文探讨了在 SQL Server 中使用 JDBC 插入数据时遇到的问题。通过详细分析代码和数据库配置,提供了解决方案并解释了潜在的原因。 ... [详细]
  • 本题要求在一组数中反复取出两个数相加,并将结果放回数组中,最终求出最小的总加法代价。这是一个经典的哈夫曼编码问题,利用贪心算法可以有效地解决。 ... [详细]
  • 主调|大侠_重温C++ ... [详细]
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社区 版权所有