一、进程空间
- MMU
蓝色标出了真实使用的内存大小
二、栈内存
- 笔记
ulimit -a
栈的大小是10M,栈不能存放大量的数据,栈是用来
数据交换
的
2.1 栈溢出
- 数组开得过大
- 递归层数过多
2.2 栈的发展方向
- 高地址往低地址发展
三、堆内存
3.1 堆的申请大小
-
从低地址到高地址
四、堆内存的申请与释放
- 在栈和堆中申请空间的对比——
4.1 申请和初始化的最小单位均是字节
- 数组类型在
栈
和堆
上
所以我们使用
memset
,最后得到的10个数并不是1,而是一个很大的数字,因为每个数是4个字节,每个字节是01
,所以就是十六进制的01010101
,也就是十进制的16843009
- 注意,
pa
本身是在栈上的,pa
申请的东西是在堆上的
4.2 小陷阱
这个程序这样写最后会导致程序崩溃,因为pa
最后会偏离原先的位置,那么之前memset
分配的40个字节就不再与现在的pa
适配,最后就会导致崩溃,所以如果要这样写,提前拷贝一份即可——
4.3 calloc的使用
- 包装了一个
memset
,自动初始化
五、realloc的使用
- 局部变量在栈上这句话是不对的
auto类型修饰的局部变量在栈上,static修饰的局部变量不在栈上
递归会爆栈的原因是会不断调用函数,调用函数就需要开辟新的空间,从函数返回才能释放掉这片空间,但是递归层数太多的时候空间一定不会释放掉,就会导致不断在开辟新的空间
malloc
和calloc
调用失败都是返回NULL
,区别是calloc
会初始化。
5.1 用于扩容的realloc
- 需要注意的问题,如果扩容的时候,
指针ptr
指向的位置后面还有多余的空间,那么ptr指向的位置不变,直接在原先空间的后面再扩容即可。但是如果原先位置后面的空间不够扩容,那么ptr
将会更换位置,重新指向一片新的空间,然后申请足够的内存来实现扩容——
比如上面的newPa
是可能和pa
不相等的
- 当然也可以这么写
5.2 free的使用
- free的使用不用指定
字节数
,因为malloc()
在使用的时候申请了多少个字节后台是有记录的——
任意+1
,-1
会导致程序崩溃
六、服务器导致的内存泄漏
七、堆内存的使用
7.1 置空很重要
申请内存,判断是否申请成功,strcpy则对应的使用,然后free释放,然后pc = NULL
则是置空——
7.2 谁申请谁释放
- 不然很容易会double free——
main()函数
中申请的就main()函数中释放
7.3 开放的地址空间
- 传
对象的地址
到不同的作用域,可依据地址修改地址所指向对象的内容,本质是地址空间是开放的
7.4 堆与栈空间的返回
-
栈上的数值和地址
是可以返回的 - 但是
栈上的空间
是不可以返回的