C++和C分别使用什么函数来做内存的分配和释放?有什么区别,能否混用?
- C使用
malloc
和free
来进行内存的分配和释放,需要指定多大的内存 - C使用
new
和delete
来进行内存的分配和释放,不需要指定多大内存,只需要声明类型。准确的说new
和delete
是C里的关键字,若是要深究是哪个函数来实现内存管理的话应该是operator new
和operator delete
这两个函数 - 想探究
new
和malloc
之间的区别,就需要解释一下C++中new
和delete
到底做了什么- 首先我们new一个类型,实际上编译器会做3个操作。
- 第一个就是调用
operator new
函数,这个函数在做的事情就是在循环调用malloc申请空间,申请成功就返回空间的首地址,是一个void指针,申请失败就会调用一个new handler,一个解决内存不足的办法,然后再循环回去看看能不能malloc成功,如果最后还是没有申请得到空间,就会抛出异常。值得一提的是,这个operator new函数是可以重载的,这就涉及到后面内存池或者分配器的设计了。 - 第一个操作是调用operator new,第二个操作就是转型了,再C中malloc回来的是一个void指针我们需要进行强制转型,在刚刚说的operator new 返回的也是一个void指针,只不过转型操作由系统自动进行了。
- 转型完成之后就是第三个操作,就是调用类的构造函数。这也是和malloc很大的不同,我们new出来的类对象系统会调用构造函数。当然这里说的调用构造函数也只有编译器能做到,我们是不能直接调用一个类的构造函数。只想调用类的构造函数而不申请空间,就是另一个placement new能做到的事了
- 第一个就是调用
- 以上是我们使用new时,编译器做的事情,而当我们使用delete销毁一个内存的时候,系统也做了两个操作。
- 第一个就是调用类对象的析构函数,刚刚说的构造函数我们不能显式调用,但析构函数我们是可以显示调用的。
- 第二个操作就是调用operator delete函数,这个函数内部就是一个free函数,释放掉对应的内存。
- 首先我们new一个类型,实际上编译器会做3个操作。
- 刚刚忘记说了,虽然我们new 一个类对象的时候不需要说明多大的空间,但其实operator new 的第一个参数还是要多少字节和malloc一个,但是编译器会计算这个类型的大小有多大,所以我们new的时候不用写申请多少空间。另外它其实还有第二个的参数是一个指针,这是placement new才会用到的东西。
nt new才会用到的东西。 - operator delete 的参数也和free一样是一个指针,指向你要释放的首地址,之所以delete和malloc为什么知道要释放首地址往后多少空间的内存,就跟这个内存头部的cookie有关了。