内存空间:
内存空间都是独立的、相互隔离的。对于每个进程来讲,看起来应该都是独占的。
内存管理
内存管理包含: 物理内存管理; 虚拟内存管理; 两者的映射(虚拟地址和物理地址如何映射)
使用内存的几种方式
简单的程序
#include <stdio.h>
#include <stdlib.h>
int max_length = 128;
char * generate(int length){
int i;
char * buffer = (char*) malloc (length+1);
if (buffer == NULL)
return NULL;
for (i=0; i<length; i++){
buffer[i]=rand()%26+'a';
}
buffer[length]='\0';
return buffer;
}
int main(int argc, char *argv[])
{
int num;
char * buffer;
printf ("Input the string length : ");
scanf ("%d", &num);
if(num > max_length){
num = max_length;
}
buffer = generate(num);
printf ("Random string is: %s\n",buffer);
free (buffer);
return 0;
}
用户部分
- 全局变量,例如 max_length
- 常量字符串"Input the string length : ";
- 函数栈:涉及了函数调用,局部变量,函数参数等都是保存在这里
- 堆,malloc 分配的内存在堆里面
- 系统调用里面涉及对 glibc 的调用, glibc 的代码会以 so 文件的形式存在的放在内存里面。
malloc 会调用系统调用,进入内核,内核部分还需要分配内存
内核部分
- 内核中也有全局变量
- 每个进程都要有一个 task_struct
- 每个进程还有一个内核栈
- 在内核里面也有动态分配的内存
- 虚拟地址到物理地址的映射 后续了解
虚拟地址
不管映射到物理地址以后是如何布局的,从虚拟的角度来看,一大片连续的内存空间就是独占的
- 32 位,有 2^32 = 4G 的内存空间
- 64 位,在 x86_64 下面,其实只使用了 48 位,48 位地址长度也就是对应了 256TB 的地址空间
这一大片连续的虚拟空间,一部分用来放内核的东西,称为内核空间,一部分用来放进程的东西,称为用户空间。用户空间在下,在低地址,内核空间在上,在高地址。
用户空间
用户空间从低到高布局为: 代码段; DATA 段; BSS 段(未初始化静态变量); 堆段; 内存映射段; 栈地址空间段。 普通进程整个空间是它独占的
- 代码段 Text Segment
存放二进制可执行代码的位置 - DATA 段 Data Segment
存放静态常量 - BSS 段(未初始化静态变量) BSS Segment
存放未初始化的静态变量 - 堆(Heap)段
堆是往高地址增长的,是用来动态分配内存的区域,malloc 就是在这里面分配的。 - 内存映射段 Memory Mapping Segment
这块地址可以用来把文件映射进内存用的,如果二进制的执行文件依赖于某个动态链接库,就是在这个区域里面将 so 文件映射到了内存中。 - 栈(Stack)地址段
主线程的函数调用的函数栈就是用这里的
如果需要进行更高权限的工作,就需要调用系统调用,进入内核
内核空间
内核空间: 多个进程看到同一内核空间, 内核空间其实是完全共享但内核栈每个进程不一样,虽然内核空间共享但作为内核代码,我都能动,但是不随便动
- 如果要访问一些公共的数据结构,需要进行锁保护
- 内核代码也仅能访问内核空间
- 内核也有内核代码段, DATA 段, 和 BSS 段;
参考资料:
趣谈Linux操作系统(极客时间)链接:
http://gk.link/a/10iXZ
欢迎大家来一起交流学习