1.C语言内存管理方式
C 语言通过标准库<stdlib.h> 提供以下函数管理堆内存:
1.1.malloc函数
void* malloc(size_t size);
在堆上分配指定大小的一块连续的空间,(以字节为单位返回该内存的其实位置,如果申请失败的话,返回NULL
1.2.calloc函数
void* calloc(size_t num, size_t size);
和malloc的区别
第一点:参数不同,malloc的参数是直接分配给予的字节数大小,calloc参数,第一个是个数,第二个是字节数
第二点:calloc为开辟的内存进行了初始化,都初始化为0,如果申请失败的话,返回NULL
1.3realloc函数
void* realloc(void* ptr, size_t size);
用于调整之间申请的空间,ptr
是之前通过 malloc
、calloc
或 realloc
分配的内存块的指针,size
是新的内存块大小。若调整成功,则返回指向新内存块的指针;若失败,则返回 NULL
,原内存块不会被释放。
分配空间的时候有两种情况:
1.原空间后面有足够的空间:
直接在后面进行申请
2.原空间后面不够我需要申请的空间:
重新在堆上找到一块足够大的空间,先将原数据复制到新找的空间,再将原空间进行释放
1.4.free函数
释放之前通过 malloc
、calloc
或 realloc
分配的内存块。ptr
是指向要释放的内存块的起始位置
注意:释放内存后,指向该内存的指针会变成悬空指针。若继续使用悬空指针,会导致未定义行为。因此,释放内存后,应将指针置为 NULL
。
2.c++内存管理方式
2.1.new delete操作内置类型
1.new和delete
new:
基本语法:
type: 是内置类型,
pointer:是指向该类型对象的指针。
new: 操作符会在堆上分配足够的内存来存储该类型的对象,并返回指向该内存位置的指针。
第二种方式:直接进行初始化将pointrt指向的值置为value
示例:
delet:
语法:
pointer:指向的指针
示例:
2.new[]和delete[]
new[]:
分配内置类型数组
第二种方式:进行初始化
示例:
在堆上分配了5个int的数组,并将首元素地址给arrPtr
delete[]:
语法:
示例:
2.2.new delete操作自定义类型
基本用法:
new 运算符用于在堆上分配内存并构造自定义类型的对象
delete运算符则用于销毁对象并释放其占用的内存。
示例:
new
运算符:new MyClass()
在堆上分配了足够的内存来存储 MyClass
对象,并调用其构造函数进行初始化。返回的是指向该对象的指针 obj
。
delete
运算符:delete obj
调用 MyClass
对象的析构函数,释放对象占用的资源,然后释放该对象在堆上的内存。
工作原理:
new
运算符:- 调用
operator new
函数来分配所需的内存空间。operator new
是一个全局函数,也可以为自定义类型重载该函数。 - 调用自定义类型的构造函数来初始化分配的内存。
- 调用
delete
运算符:- 调用自定义类型的析构函数来清理对象占用的资源。
- 调用
operator delete
函数来释放之前分配的内存。
数组操作:
new[]
运算符:new MyClass[3]
在堆上分配了能存储 3 个MyClass
对象的连续内存空间,并依次调用每个对象的构造函数进行初始化
delete[]
运算符:delete[] arr
会依次调用数组中每个对象的析构函数,然后释放整个数组占用的内存。需要注意的是,使用new[]
分配的内存必须使用delete[]
来释放,否则会导致内存泄漏和未定义行为。
异常处理:
异常捕获:在 try
块中使用 new
运算符分配大量内存,如果内存不足,会抛出 std::bad_alloc
异常。在 catch
块中捕获该异常,并输出错误信息。
2.3.new 和delete实现原理
1.new 操作符实现流程:
第一步:调用operator new函数在堆上申请空间
第二步:调用类的构造函数,对类进行初始化
2.delete操作符实现流程:
第一步:调用类中的析构函数
第二步:调用 operator delete函数,释放之前在堆上申请的空间
3.operator new和operator delete函数
operator new:
是一个全局函数,用于在堆上分配指定大小的内存块。当使用new 运算符创建对象时,实际上会先调用 operator new函数来分配内存,然后再调用对象的构造函数进行初始化。
函数原型:
operator delete:
是一个全局函数,用于释放之前由operator new分配的内存块。当使用delete运算符销毁对象时,实际上会先调用对象的析构函数进行清理,然后再调用operator delete函数释放内存。
函数原型:
示例:
手动分配内存:
operator new(sizeof(MyClass))
手动调用 operator new
函数分配了足够存储 MyClass
对象的内存块。
定位 new
:new (rawMemory) MyClass()
使用定位 new
在分配的内存上构造 MyClass
对象。
显式调用析构函数:
obj->~MyClass()
显式调用 MyClass
对象的析构函数。
手动释放内存:
operator delete(rawMemory)
手动调用 operator delete
函数释放之前分配的内存。
重载operator new和operator delete函数
可以在重载的函数体中自定义要发送的信息,重载函数是在类当中进行重载