内存管理基础
直接使用物理内存的问题
- 地址空间不隔离
- 内存使用低效
程序运行地址不确定
=>添加中间层虚拟地址–>映射–>物理地址
地址空间
- 32位系统物理地址=32条总线=4GB
- 32位系统虚拟地址=默认=1.5*物理地址
- 每个进程都有自己独立的虚拟空间
虚拟地址->物理地址的映射
分段
将一段与程序所需的内存空间大小的虚拟空间映射到某个内存地址空间
- 解决了:地址空间不隔离+程序运行地址不确定
分页
将地址空间人为的等分为固定大小的页(4KB)
- 虚拟页的存储需要MMU的支持
编译和链接
预处理—编译—汇编—链接
目标文件和可执行文件
编译源代码后的目标文件( *.o 文件 )
可执行文件( windows=.exe; linux=.ELF)
可执行文件
代码段+数据段+未初始化数据段
代码段(text): 存放CPU执行的指令
- 代码段通常可以共享
- 规划了局部变量
- 代码段指令包括:操作码和操作对象
- 若操作对象是立即数(数值),则包括在代码段
- 若操作对象是,局部数据,则在运行时在栈区分配空间,然后引用数据地址
- 若是BBS和数据段,则在代码段中引用其地址
数据段(data):全局初始化数据区和静态数据区
- 全局变量,静态变量(初始化的全局静态变量和局部静态变量)
- 数据常量(字符串常量)
- static a=100; 若初始化在任何函数体外,则是全局静态变量
- static a=100; 若初始化在某函数体内,则是局部静态变量
若在函数前加static,则该函数只可以在当前文件被调用
- 未初始化数据区(bbs):存入全局未初始化的全局变量和未初始化的静态变量
- bbs区被内核初始化为0或者NULL
静态链接
多个可执行文件的链接
可执行文件的装载与进程
进程虚拟地址空间
博客:http://www.cnblogs.com/dyllove98/archive/2013/07/05/3174341.html
虚拟地址=虚拟地址空间是指程序访问的地址,也不是实际物理地址,而是OS虚拟出来的地址,OS帮你做映射等
虚拟内存=是相对物理内存,对程序来说就是访问虚拟内存
- 进程和程序:程序=静态概念,预先编译好的指令和数据集合;进程=动态概念,程序运行的过程
- 进程拥有独立的虚拟地址空间(由cpu的位数,32位=4GB)
- c语言指针大小位数=虚拟空间位数
- linux系统的虚拟空间=1GB的操作系统地址空间+3GB的用户地址空间
装载
- 覆盖装载和页映射(页映射=虚拟存储机制一部分)
- 页映射:将内存的所胡和指令分为页
操作系统角度的可执行文件装载
进程的建立
- 创建虚拟地址空间
- 将cpu指令寄存器设置为可执行文件的入口,启动运行
进程的虚拟空间分布
将相同权限的段,河北当做一个段进行映射
堆栈
- 堆:最大空间是3GB(Windows=2GB)
对齐
栈