知识点回顾
1. C语言的内存布局规律
可以看到局部变量的地址是占据高地,接着是malloc函数的动态内存空间,然后是全局变量和静态局部变量。
不过这两者都需要区分是否有已经初始化,已经初始化的放一块,未初始化的放一块,并且未初始化的地址要比初始化的要更高一些。接着下来是字符串常量,最后是函数的地址。
2. 典型的C语言程序的内存空间划分
根据内存地址从低到高分别划分为:
- 代码段(Text segment)
- 数据段(Initialized data segment)
- BSS段(Bss segment/Unintialized data segment)
- 栈(Stack)
- 堆(Heap)
3. 代码段(Text segment)
代码段通常指用来存放程序执行代码的一块内存区域。
这部分区域的大小在程序运行前就已经确定,并且内存区域通常属于只读。
在代码段中,也可能包含一些只读的常数变量,例如字符串常量。
4. 数据段(data segment)
数据段通常用来存放已经初始化的全局变量和局部静态变量。
5. BSS段(Bss segment/Uninitialized data segment)
BSS段通常是指用来存放程序中未初始化的全局变量的一块内存区域。
BSS的英文是Block Started by Symbol的简称,这个区段中的数据在程序运行前将被自动初始化为数字0.
6. 堆
前面我们学习了动态内存管理函数,使用他们申请的内存空间就是分配在这个堆里边。
所以堆是用于存放进程中被动态分配的内存段,它的大小并不固定,可动态扩展或缩小。
7. 栈
栈 是函数执行的内存区域,通常和堆共享一片区域。
8. 堆和栈的区别
堆和栈则是C语言运行的最重要的元素,下面我们将其对比
申请方式
- 堆是由程序员手动申请
- 栈是有系统自动分配
释放方式 - 堆是由程序员手动释放
- 栈是有系统自动释放
生存周期
堆的生存周期由动态申请到程序员主动释放为止,不同函数之间局可以自由访问。
栈的生存周期由函数调用开始到函数返回时结束,函数之间的局部变量不能互相访问。
发展方向
堆和其他区段一样,都是从低地址向高地址。
栈则相反,是由高地址向低地址。
测试题
0. 根据 C 语言的内存布局规律,通常局部变量和全局变量哪一个的地址更小?
答:全局变量的地址更小。
1. 下面代码中,为何两个不同的变量可以存放在同一个地址上?
#include <stdio.h>
void func1(void);
void func2(void);
void func1(void)
{
int a = 520;
printf("value of a is %d\n", a);
printf("addr of a is %p\n", &a);
}
void func2(void)
{
int b = 880;
printf("value of b is %d\n&#