new和malloc的区别
参考 http://www.cnblogs.com/fly1988happy/archive/2012/04/26/2470542.html
参考 http://www.cnblogs.com/fly1988happy/archive/2012/04/16/2452021.html
文章目录
1. malloc()函数
malloc 的全称是 memory allocation,中文叫动态内存分配。
原型:extern void *malloc(unsigned int num_bytes);
说明:分配长度为 num_bytes 字节的内存块。如果分配成功则返回指向被分配内存的指针,分配失败返回空指针 NULL。当内存不再使用时,应使用 free() 函数将内存块释放。
返回类型是 void* 类型。void* 表示未确定类型的指针。C,C++ 规定,void* 类型可以强制转换为任何其它类型的指针。
void* 表示未确定类型的指针,更明确的说是指申请内存空间时还不知道用户是用这段空间来存储什么类型的数据(比如是 char 还是 int 或者…)
1.1 malloc与free
void free(void *FirstByte);
该函数是将之前用 malloc 分配的空间还给程序或者是操作系统,也就是释放了这块内存,让它重新得到自由,可以被重新分配和使用。
1.2 注意事项
1)申请了内存空间后,必须 检查是否分配成功。
2)当不需要再使用申请的内存时,记得释放;释放后应该把指向这块内存的指针 指向NULL,防止程序后面不小心使用了它。
3)这两个函数应该是配对。如果申请后不释放就是内存泄露;如果无故释放那就是什么也没有做。释放只能一次,如果释放两次及两次以上会出现错误(释放空指针例外,释放空指针其实也等于啥也没做,所以释放空指针释放多少次都没有问题)。
4)虽然 malloc() 函数的类型是 void * ,任何类型的指针都可以转换成 void * ,但是最好还是在前面进行 强制类型转换,因为这样可以躲过一些编译器的检查。
1.3 malloc()到底从哪里得到了内存空间?
答案是从堆里面获得空间。也就是说函数返回的指针是 指向堆里面的一块内存。
操作系统中有一个 记录空闲内存地址的链表。当操作系统收到程序的申请时,就会遍历该链表,然后就寻找第一个空间大于所申请空间的堆结点,然后就将该结点从空闲结点链表中删除,并将该结点的空间分配给程序。
2. new运算符
2.1 C++中,用new和delete动态创建和释放数组或单个对象。
动态创建对象时,只需指定其数据类型,而不必为该对象命名,new表达式返回指向该新创建对象的指针,我们可以通过指针来访问此对象。
int *pi = new int;
这个 new 表达式在堆区中分配创建了一个整型对象,并返回此对象的地址,并用该地址初始化指针 pi 。
2.2 动态创建对象的初始化
动态创建的对象可以用初始化变量的方式初始化。
int *pi=new int(100);
//指针pi所指向的对象初始化为100
string *ps=new string(10,’9’);//*ps 为“9999999999”
如果不提供显示初始化,对于类类型,用该类的默认构造函数初始化;而内置类型的对象则无初始化。
也可以对动态创建的对象做值初始化:
int *pi=new int( );
//初始化为0
int *pi=new int;//pi 指向一个没有初始化的int
string *ps=new string( );//初始化为空字符串 (对于提供了默认构造函数的类类型,没有必要对其对象进行值初始化)
2.3 撤销动态创建的对象
delete 表达式释放指针指向的地址空间。
delete pi ;
// 释放单个对象
delete [ ] pi;//释放数组
如果指针指向的不是 new 分配的内存地址,则使用 delete 是不合法的。
2.4 在delete之后,重设指针的值
delete p; //执行完该语句后,p 变成了不确定的指针,在很多机器上,尽管 p 值没有明确定义,但仍然存放了它之前所指对象的地址,然后 p 所指向的内存已经被释放了,所以 p 不再有效。此时,该指针变成了 悬垂指针(悬垂指针指向曾经存放对象的内存,但该对象已经不存在了)。悬垂指针往往导致程序错误,而且很难检测出来。
一旦删除了指针所指的对象,立即将指针置为 0,这样就非常清楚的指明指针不再指向任何对象。(零值指针:int *ip = 0;)
2.5 new分配失败时,返回什么?
1993 年前,c++ 一直要求在内存 分配失败时 operator new 要返回 0,现在则是要求 operator new抛出 std::bad_alloc 异常。很多 c++ 程序是在编译器开始支持新规范前写的。c++ 标准委员会不想放弃那些已有的遵循返回 0 规范的代码,所以他们提供了另外形式的 operator new ( 以及operator new[] )以继续提供返回 0 功能。这些形式被称为“无抛出”,因为他们没用过一个throw,而是在使用new的入口点采用了nothrow 对象:
class widget { … };
widget *pw1 = new widget;// 分配失败抛出std::bad_alloc
if (pw1 == 0) … // 这个检查一定失败
widget *pw2 = new (nothrow) widget; // 若分配失败返回0
if (pw2 == 0) … // 这个检查可能会成功
3. malloc和new的区别
new 返回指定类型的指针,并且可以自动计算所需要大小。
int p;
p = new int;//返回类型为 int* 类型(整数型指针),分配大小为 sizeof(int);
或:
int parr;
parr = new int [100];//返回类型为 int* 类型(整数型指针),分配大小为 sizeof(int) * 100;
而 malloc 则必须要由我们计算字节数,并且在返回后强行转换为实际类型的指针。
int* p;
p = (int *) malloc ( sizeof(int)*128 );//分配128个(可根据实际需要替换该数值)整型存储单元,并将这128个连续的整型存储单元的首地址存储到指针变量p中
double *pd = (double *) malloc (sizeof(double)*12); //分配12个double型存储单元,并将首地址存储到指针变量pd中
malloc 只管分配内存,并不能对所得的内存进行初始化,所以得到的一片新内存中,其值将是随机的。
除了分配及最后释放的方法不一样以外,通过 malloc 或 new 得到指针,在其它操作上保持一致。
4. 有了malloc/free为什么还要new/delete?
-
malloc 与 free 是 C++/C 语言的标准库函数,new/delete 是C++的运算符。它们都可用于申请动态内存和释放内存。
-
对于 非内部数据类型的对象 而言,光用 maloc/free 无法满足动态对象的要求。对象在创建的同时要自动执行构造函数,对象在消亡之前要自动执行析构函数。由于 malloc/free 是库函数而不是运算符,不在编译器控制权限之内,不能够把执行构造函数和析构函数的任务强加于 malloc/free。
因此 C++ 语言需要 一个能完成动态内存分配和初始化工作的运算符 new,以及 一个能完成清理与释放内存工作的运算符delete。
我们不要企图用 malloc/free 来完成动态对象的内存管理,应该用 new/delete。由于内部数据类型(int、float)没有构造与析构的过程,对它们而言 malloc/free 和 new/delete 是等价的。
- 既然 new/delete 的功能完全覆盖了 malloc/free,为什么 C++ 不把 malloc/free 淘汰出局呢?这是因为 C++ 程序经常要调用 C 函数,而 C 程序只能用 malloc/free 管理动态内存。
如果用 free 释放 new 创建的动态对象,那么该对象因无法执行析构函数而可能导致程序出错。如果用 delete 释放 malloc 申请的动态内存,结果也会导致程序出错,同时该程序的可读性很差。
所以 new/delete 必须配对使用,malloc/free 也一样。