以Linux下的32位环境为例子
内核空间和用户空间
32位环境理论有4g的虚拟地址空间,其中1G交给操作系统内核使用(内核空间),剩下3G交给用户使用(用户空间)
经典内存模型
内存分区 | 说明 |
---|---|
程序代码区 | 存放函数体的二进制代码。 |
常量区 | 常量,字符串常量,只能读不能写。 |
全局数据区 | 全局变量静态变量,读写,。 |
堆区 | 有程序员申请释放,若不释放,程序结束后,操作系统回收 |
栈区 | 存放函数的参数值,局部变量等 |
程序代码区、常量区、全局数据区在程序加载到内存后就分配好了,并且在程序运行期间一直存在,不能销毁也不能增加(大小已被固定),只能等到程序运行结束后由操作系统收回,所以全局变量、字符串常量等在程序的任何地方都能访问,因为它们的内存一直都在。
栈区:程序代码区、常量区、全局数据区在程序加载到内存后就分配好了,并且在程序运行期间一直存在,不能销毁也不能增加(大小已被固定),只能等到程序运行结束后由操作系统收回,所以全局变量、字符串常量等在程序的任何地方都能访问,因为它们的内存一直都在。
堆区:程序员唯一能控制的内存区域,它是一块巨大的内存空间,常常占据整个虚拟空间的绝大部分,在这片空间中,程序可以申请一块内存,并自由地使用(放入任何数据)。堆内存在程序主动释放之前会一直存在,不随函数的结束而失效。在函数内部产生的数据只要放到堆中,就可以在函数外部使用。
#include <stdio.h>
char *str1 = "c.biancheng.net"; //字符串在常量区,str1在全局数据区
int n; //全局数据区
char* func(){
char *str = "C语言中文网"; //字符串在常量区,str在栈区
return str;
}
int main(){
int a; //栈区
char *str2 = "01234"; //字符串在常量区,str2在栈区
char arr[20] = "56789"; //字符串和arr都在栈区
char *pstr = func(); //栈区
int b; //栈区
}
- 全局变量的内存在编译的时候就分配好了,默认值为0
- 函数 func() 中的局部字符串常量"C语言中文网"也被存储到常量区,不会随着 func() 的运行结束而销毁,所以最后依然能够输出。
- 字符数组 arr[20] 在栈区分配内存,字符串"56789"就保存在这块内存中,而不是在常量区。