C/C++/linux内存分布
语言 | linux |
---|---|
内核 | 内核 |
栈 | 栈 |
内存映射段 | 内存映射段 |
堆 | 堆 |
全局数据区 | 数据段 |
静态数据区 | 数据段 |
文字常量区 | 代码段 |
代码段 | 代码段 |
栈: 局部变量
堆: 动态开辟的空间
文字常量区: 字符串
代码段: 二进制数
数组名: 数组首元素的地址
特殊:
sizeof,&数组名: 代表整个数组
sizeof: 类型的大小
strlen: 字符串有效字符个数,看/0结束的位置,通常大小为sizeof+1
malloc、calloc、realloc
malloc: 只申请空间,不初始化
int* pa = (int*)malloc(sizeof(int));
calloc: 申请空间,初始化为0
int num = 2;
int* pb = (int*)calloc(num,sizeof(int));
realloc: 重新申请已有空间(调整空间的首地址和大小,如果原空间有内容,则把内容拷贝到新的空间)
int* pc = (int*)realloc(pa,sizeof(int)*100);
realloc的三种逻辑
第一种:(最常见)
1. 重新开空间
2. 拷贝原有内容
3. 释放原有空间
int* pc = (int*)realloc(pa,sizeof(int)*2);
第二种:
申请新的空间(修改头信息)
char* pa = (char*)malloc(sizeof(char));
*pa = 'a';
char* pc = (char*)realloc(pa,sizeof(char)*2);
第三种:
减小空间大小
char* pa = (char*)malloc(sizeof(char)); //1 byte
*pa = 'a';
char* pc = (char*)realloc(pa,sizeof(char)*2); //2 byte
char* pd = (char*)realloc(pc,sizeof(char)); //1byte
realloc(nullptr,字节数):功能等价于malloc
char* pe = (char*)realloc(nullptr,sizeof(char));
new/delete
C++内存管理方式: 通过new和delete操作符进行动态内存管理。
new/delete与malloc/free用法相似,但仍有不同。
new在申请和释放内存时会自动调用对象的构造函数和析构函数,完成初始化;
malloc则只管分配内存,并不对所得的内存初始化。
new也可用于自定义类型。
free不会调用析构,delete/delete[ ] 调用一次/多次析构。
new: 动态开辟空间,构造
delete: 析构,释放空间
new --> delete
malloc --> free/delete
new[ ] --> delete[ ]
在堆上申请4个byte空间
int* pa = new int;
在堆上申请连续的空间
int* pb = (int*)malloc(sizeof(int)*10);
int*pb2 = new int;
在堆上申请空间+初始化
int* pc = new int(20);
释放一个元素的空间
free (pa) ;
delete pa2 ;
释放连续的空间
free (pb) ;
delete [ ] pb2 ;
operator new与operator delete函数
operator new
不是new的运算符重载函数,而是全局函数,使用方式类似于malloc。
内部通过malloc来申请空间,当malloc申请空间成功时直接返回;当申请空间失败时,尝试执行空间不足应对措施,如果用户设置了应对措施,则继续申请,否则抛异常。
void *__CRTDECL operator new(size_t size) _THROW1(_STD bad_alloc) {
// try to allocate size bytes
void *p;
while ((p = malloc(size)) == 0)
if (_callnewh(size) == 0)
{
// report no memory
// 如果申请内存失败了,这里会抛出bad_alloc 类型异常
static const std::bad_alloc nomem;
_RAISE(nomem);
}
return (p);
}
operator delete
最终通过free来释放空间。
封装free,不会抛异常。
malloc/free和new/delete的区别
相同点: 都是从堆上申请空间,且都需要用户手动释放。
不同点:
1. malloc和free是函数;new/delete是操作符。
2. malloc申请的空间不会初始化;new会初始化。
3. malloc申请空间时,需要手动计算空间大小并传递;new只需要在其后跟上空间的类型即可。
4. malloc的返回值为void*,在使用时必须强转;new不需要,因为new后面跟的时空间的类型。
5. malloc申请空间失败时,返回的是NULL,因此使用时必须判空;new不需要,但是new需要捕获异常。
6. 申请自定义对象时,malloc和free只会开辟空间,不会调用构造函数和析构函数;new在申请空间后会调用构造函数完成对象的初始化,delete在释放空间时前会调用析构函数完成空间中的资源清理。