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




推荐阅读
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&# ... [详细]
  • 本文详细探讨了VxWorks操作系统中双向链表和环形缓冲区的实现原理及使用方法,通过具体示例代码加深理解。 ... [详细]
  • Codeforces Round #566 (Div. 2) A~F个人题解
    Dashboard-CodeforcesRound#566(Div.2)-CodeforcesA.FillingShapes题意:给你一个的表格,你 ... [详细]
  • 在多线程编程环境中,线程之间共享全局变量可能导致数据竞争和不一致性。为了解决这一问题,Linux提供了线程局部存储(TLS),使每个线程可以拥有独立的变量副本,确保线程间的数据隔离与安全。 ... [详细]
  • 本文介绍如何使用Objective-C结合dispatch库进行并发编程,以提高素数计数任务的效率。通过对比纯C代码与引入并发机制后的代码,展示dispatch库的强大功能。 ... [详细]
  • 本实验主要探讨了二叉排序树(BST)的基本操作,包括创建、查找和删除节点。通过具体实例和代码实现,详细介绍了如何使用递归和非递归方法进行关键字查找,并展示了删除特定节点后的树结构变化。 ... [详细]
  • 文件描述符、文件句柄与打开文件之间的关联解析
    本文详细探讨了文件描述符、文件句柄和打开文件之间的关系,通过具体示例解释了它们在操作系统中的作用及其相互影响。 ... [详细]
  • 本文详细介绍了C语言中链表的两种动态创建方法——头插法和尾插法,包括具体的实现代码和运行示例。通过这些内容,读者可以更好地理解和掌握链表的基本操作。 ... [详细]
  • Linux设备驱动程序:异步时间操作与调度机制
    本文介绍了Linux内核中的几种异步延迟操作方法,包括内核定时器、tasklet机制和工作队列。这些机制允许在未来的某个时间点执行任务,而无需阻塞当前线程,从而提高系统的响应性和效率。 ... [详细]
  • 本文探讨了 Objective-C 中的一些重要语法特性,包括 goto 语句、块(block)的使用、访问修饰符以及属性管理等。通过实例代码和详细解释,帮助开发者更好地理解和应用这些特性。 ... [详细]
  • C++构造函数与初始化列表详解
    本文深入探讨了C++中构造函数的初始化列表,包括赋值与初始化的区别、初始化列表的使用规则、静态成员初始化等内容。通过实例和调试证明,详细解释了初始化列表在对象创建时的重要性。 ... [详细]
  • 本题通过将每个矩形视为一个节点,根据其相对位置构建拓扑图,并利用深度优先搜索(DFS)或状态压缩动态规划(DP)求解最小涂色次数。本文详细解析了该问题的建模思路与算法实现。 ... [详细]
  • 实体映射最强工具类:MapStruct真香 ... [详细]
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社区 版权所有