C语言内存四区的学习总结(一)---- 静态区
最近重新学习C语言相关知识,重新提到内存四区的概念,那么在之前的学习的基础上,在这儿做一个简单的总结与分享。
一、内存四区建立的流程
可以简单直观的查看下面的这个图片,直接的说明我们的程序在内存中是如何去存储,运行。。。。

程序运行的流程说明
1、操作系统把物理硬盘代码 load到内存
2、操作系统把c代码分成四个区
3、操作系统找到 main函数入口执行
二、各区元素分析
栈区(stack):由编译器自动分配和释放,存放函数的参数值,局部变量的等临时的值.
堆区(heap):一般由程序员分配释放(动态内存申请与释放),若程序员不释放,程序结束时可能由操作系统回收
全局区(静态区) (static):全局变量和静态变量的存储是放在一块的,初始化的全局变量和静态变量在一块区域,未初始化的全局变量和未初始化的静态变量在相邻的另一块区城, 该区域在程序结束后由操作系统释放
常量区:字符串常量和其他常量的存储位置,程序结束后由操作系统释放
程序代码区:存放函教体的二进制代码
三、静态存储区和栈区的分析
就下面的程序做一个简单的分析:
#include "stdio.h" char *getStr1()
{
char *p1 = "abcd1";
return p1;
} char *getStr2()
{
char *p2 = "abcd2";
return p2;
} char *getStr3()
{
char *p3 = "abcd2";
return p3;
} int main(int argc, const char **argv)
{
char *p1 = NULL;
char *p2 = NULL;
char *p3 = NULL; p1 = getStr1();
p2 = getStr2();
p3 = getStr3(); printf("p1 = %s, p2 = %s, p3 = %s\r\n", p1, p2, p3);
printf("p1 = %d, p2 = %d, p3 = %d\r\n", p1, p2, p3); return ;
}
可以得出函数调用的模型为:

四、根据上面的程序开始逐步进行分析
在main开始执行之前,先进行假设栈区的开口向上,栈区中的横线形象的比方函数之间分割线,那么,在main开始执行的时候,
1、先定义了三个指针变量,分别是p1、p2、p3;那么此时在栈区会进栈三个变量,依次是p1 --> p2 --> p3;
2、调用函数getStr1,此时进栈p1;注意,此处的p1和main函数中的p1不是一个p1;
3、在常量区定义字符串”abcd1”,然后getStr1中的p1指针指向这个字符串(图中①,为黑色),假设字符串”abcd1”的首地址为0xaabb;
4、函数getStr1返回,将getStr1中的指针p1析构(图中用灰色表示,表示已经不存在了),所以①号线也随之断裂(灰色表示),将字符串”abcd1”的地址赋给main函数中的p1(图中②,地址用紫色表示)
5、调用函数getStr2,此时进栈p2,此处的p2和main函数中的p2不是一个p2
6、在常量区定义字符串”abcd2”,然后getStr2中的p2指针指向这个字符串(图中③,为橙色),假设字符串”abcd2”的首地址为0xbbcc;
7、函数getStr2返回,将getStr2中的指针p2析构(图中用浅橙色表示,表示已经不存在了),所以③号线也随之断裂(浅橙色表示),将字符串”abcd1”的地址赋给main函数中的p2(图中④,地址用橙色表示)
8、调用函数getStr3,此时进栈p3,此处的p3和main函数中的p3不是一个p3
9、此时本应该在常量区定义字符串” abcd2”,但是因为getStr3中的字符串和getStr2中的字符串是一样的,所以代码在编译的时候,编译器已经优化了并且只保留一个字符串”abcd2”,所以将getStr3中的p3指针指向这个字符串(图中⑤,为天蓝色)
10、函数getStr3返回,将getStr3中的指针p3析构(图中用浅天蓝色表示,表示已经不存在了),所以⑤号线也随之断裂(浅天蓝色表示),将字符串”abcd2”的地址赋给main函数中的p3(图中⑥,地址用橙色表示),此时,main函数中p2和p3实际上指向同一个内存。
至此,getStrx(x=1,2,3)已经依次执行完毕,其中进栈的变量(p1,p2,p3均已经被析构),而main函数中的p1,p2,p3均从NULL指向属于自己的内存地址。

所以,在main函数中打印输出字符串的时候,可以看出效果为:

五、简单的总结
1、因为在程序运行过程中,系统将代码区代码load到内存中,所以,代码区透明的,此后不需要关注
2、指针指向谁,就将谁的地址赋给指针:char *p1 = “abcd1”;
3、指针变量和他指向的内存空间变量是两个不同的概念p1,”abcd1”
4、不同的局部变量(不同函数中)的常量(比如字符串常量“abcd2”),如果常量相同的话编译器会进行优化,讲这些一样的常量划分到一个全局区的某个地址中,只不过用参数的形式可以通过函数调用对此参数进行初始化。
C语言内存四区的学习总结(一)---- 静态区的更多相关文章
- C语言内存四区的学习总结(三)---- 栈区
接上篇内存四区的堆区的总结,下面做一些栈区的相关总结. 一.栈区的分析: 就下面测试程序 #include "stdio.h" #include "string.h&qu ...
- C语言内存四区的学习总结(二)---- 堆区
接上篇,内存四区的分析-静态区,下面来说明一下堆区总结. 堆区分析: 堆区(heap):一般由程序员分配释放(动态内存申请与释放),若程序员不释放,程序结束时可能由操作系统回 就下面的程序: #inc ...
- C语言内存四区
按照老版操作系统来学习,内存对于程序来讲分四区.分别是 代码区,静态区,栈,堆. 由上面程序执行的结果可知: 貌似结果就是 静态代码堆栈 静态区存放的是程序中所有静态变量和常量的值.静态区的大小是程序 ...
- c语言内存四区、数据存储范围和内存存储方向
(1)代码区通常是共享只读(代码无法修改)的,即可以被其他的程序调用,例如运行两个qq,除了数据不一样,代码都是一样的, 每次运行qq,都会将代码和数据加载到内存中,除了数据,每次加载的代码都是一样的 ...
- 浅谈C语言内存管理、内存泄露、堆栈
1.内存分配区间: 对于一个C语言程序而言,内存空间主要由五个部分组成:代码段(.text).数据段(.data).静态区(.BSS).堆和栈组成. BSS段:BSS段 ...
- C语言程序猿必会的内存四区及经典面试题解析
前言: 为啥叫C语言程序猿必会呢?因为特别重要,学习C语言不知道内存分区,对很多问题你很难解释,如经典的:传值传地址,前者不能改变实参,后者可以,知道为什么?还有经典面试题如下: #include & ...
- C语言提高 (1) 第一天 数据类型本质与内存四区
(物联网的分层的概念 b/s c/s 结构 习惯: 在C语言 0 函数执行成功 <0是错误 >1做一些返回值处理 3 课前准备 工作经验,记录 4 数据类型的本质 数据类型的本质是固定大小 ...
- 深入理解C语言-深入理解内存四区
数组与指针 当数组做函数参数的时候,会退化为一个指针 此时在函数内是得不到数组大小的 因此,数组做函数参数的时候需要传递数组大小,也就是多传递一个参数 void func(int arr[], int ...
- C语言进阶之路(一)----C语言的内存四区模型
内存四区模型:操作系统给C/C++编写的程序分配内存,通常将分配的内存划分为以下四个区域:1.栈区:存放局部变量,用完由操作系统自动释放2.堆区:动态分配给程序的内存区域,由程序员手动释放3.数据区: ...
随机推荐
- 回溯法 Generate Parentheses,第二次总结
class Solution { public: vector<string> ans; void helper(string& cur, int left, int right, ...
- 尚硅谷springboot学习36-自定义starter
自定义一个starter要引一个依赖,即我们自己写的自动配置,在这个自动配置里写我们的自动配置类,属性类等 自动配置类开始类似这样 @Configuration //指定这个类是一个配置类 @Cond ...
- AHA高级心血管生命支持ACLS课前自我评估测试
AHA高级心血管生命支持ACLS课前自我评估测试 答题人:a 成绩单 您的得分:17.5 分 答对题数:7 题 您的名次:47 名 问卷满分:100 分 测试题数:40 题 参与人次:47 人 ...
- 中兴iptv机顶盒破解教程图文:亲测中兴B760EV3、B860A、B860AV1.1完美安装应用!非ttl破解![转]
一直以为中兴的这几个盒子只能通过ttl来破解,不过现在再也不用这么麻烦了,有了这个工具,前后破解不超3分钟!理论上支持所有中兴的iptv机顶盒的破解! 亲测中兴B760EV3.B860A.B860AV ...
- Powser Design 16.5 导入Mysql数据库的bug
在Power Designer 16.5中,想导入mysql的某个数据库,选择导入后会导入全部数据库. 解决方案: 打开powerdesigner.选择file--->reverse engin ...
- https://github.com/gaoyangxiaozhu/DockerVI
[更新]分享一个开源项目DockerVI,一个基于NodeJS实现的Docker Swarm可视化解决方案
- tensorflow_目标识别object_detection_api,RuntimeError: main thread is not in main loop,fig = plt.figure(frameon=False)_tkinter.TclError: no display name and no $DISPLAY environment variable
最近在使用目标识别api,但是报错了: File "/usr/local/lib/python2.7/dist-packages/tensorflow/python/ops/script_o ...
- 初学python---排序
1.永久性排序 sort() a = [12,45,1,25,3] a.sort() print(a) ----[1, 3, 12, 25, 45] 2.临时排序 sorted() a = [12, ...
- js判断访问浏览器是安卓还是ios还是微信浏览器还是微博
var u = navigator.userAgent, app = navigator.appVersion; var isAndroid = u.indexOf('Android') > - ...
- map基本方法
添加功能: V put(K key, V value) 添加和修改 ,添加时返回null,修改时返回被修改的值 Map<String,String> map = new HashMa ...