new和malloc的区别深入解析
原型:externvoid*malloc(unsignedintnum_bytes);
说明:分配长度为num_bytes字节的内存块。如果分配成功则返回指向被分配内存的指针,分配失败返回空指针NULL。当内存不再使用时,应使用free()函数将内存块释放。
说明:malloc向系统申请分配指定size个字节的内存空间,返回类型是void*类型。void*表示未确定类型的指针。C,C++规定,void*类型可以强制转换为任何其它类型的指针。
备注:void*表示未确定类型的指针,更明确的说是指申请内存空间时还不知道用户是用这段空间来存储什么类型的数据(比如是char还是int或者...)
voidfree(void*FirstByte):该函数是将之前用malloc分配的空间还给程序或者是操作系统,也就是释放了这块内存,让它重新得到自由。
1)申请了内存空间后,必须检查是否分配成功。
2)当不需要再使用申请的内存时,记得释放;释放后应该把指向这块内存的指针指向NULL,防止程序后面不小心使用了它。
3)这两个函数应该是配对。如果申请后不释放就是内存泄露;如果无故释放那就是什么也没有做。释放只能一次,如果释放两次及两次以上会出现错误(释放空指针例外,释放空指针其实也等于啥也没做,所以释放空指针释放多少次都没有问题)。
4)虽然malloc()函数的类型是(void*),任何类型的指针都可以转换成(void*),但是最好还是在前面进行强制类型转换,因为这样可以躲过一些编译器的检查。
答案是从堆里面获得空间。也就是说函数返回的指针是指向堆里面的一块内存。操作系统中有一个记录空闲内存地址的链表。当操作系统收到程序的申请时,就会遍历该链表,然后就寻找第一个空间大于所申请空间的堆结点,然后就将该结点从空闲结点链表中删除,并将该结点的空间分配给程序。
动态创建对象时,只需指定其数据类型,而不必为该对象命名,new表达式返回指向该新创建对象的指针,我们可以通过指针来访问此对象。
int*pi=newint;
这个new表达式在堆区中分配创建了一个整型对象,并返回此对象的地址,并用该地址初始化指针pi。
动态创建的对象可以用初始化变量的方式初始化。
int*pi=newint(100);//指针pi所指向的对象初始化为100
string*ps=newstring(10,"9");//*ps为“9999999999”
如果不提供显示初始化,对于类类型,用该类的默认构造函数初始化;而内置类型的对象则无初始化。
也可以对动态创建的对象做值初始化:
int*pi=newint();//初始化为0
int*pi=newint;//pi指向一个没有初始化的int
string*ps=newstring();//初始化为空字符串(对于提供了默认构造函数的类类型,没有必要对其对象进行值初始化)
delete表达式释放指针指向的地址空间。
deletepi;//释放单个对象
delete[]pi;//释放数组
如果指针指向的不是new分配的内存地址,则使用delete是不合法的。
deletep;//执行完该语句后,p变成了不确定的指针,在很多机器上,尽管p值没有明确定义,但仍然存放了它之前所指对象的地址,然后p所指向的内存已经被释放了,所以p不再有效。此时,该指针变成了悬垂指针(悬垂指针指向曾经存放对象的内存,但该对象已经不存在了)。悬垂指针往往导致程序错误,而且很难检测出来。
一旦删除了指针所指的对象,立即将指针置为0,这样就非常清楚的指明指针不再指向任何对象。(零值指针:int*ip=0;)
零值指针,是值是0的指针,可以是任何一种指针类型,可以是通用变体类型void*也可以是char*,int*等等。
空指针,其实空指针只是一种编程概念,就如一个容器可能有空和非空两种基本状态,而在非空时可能里面存储了一个数值是0,因此空指针是人为认为的指针不提供任何地址讯息。
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) ...// 这个检查可能会成功
比如:
1)int*p;
p=newint;//返回类型为int*类型(整数型指针),分配大小为sizeof(int);
或:
int*parr;
parr=newint[100];//返回类型为int*类型(整数型指针),分配大小为sizeof(int)*100;
2)而malloc则必须要由我们计算字节数,并且在返回后强行转换为实际类型的指针。
int*p;
p=(int*)malloc(sizeof(int)*128);//分配128个(可根据实际需要替换该数值)整型存储单元,并将这128个连续的整型存储单元的首地址存储到指针变量p中
double*pd=(double*)malloc(sizeof(double)*12);//分配12个double型存储单元,并将首地址存储到指针变量pd中
除了分配及最后释放的方法不一样以外,通过malloc或new得到指针,在其它操作上保持一致。
1)
2)对于非内部数据类型的对象而言,光用maloc/free无法满足动态对象的要求。对象在创建的同时要自动执行构造函数,对象在消亡之前要自动执行析构函数。由于malloc/free是库函数而不是运算符,不在编译器控制权限之内,不能够把执行构造函数和析构函数的任务强加于malloc/free。
因此C++语言需要一个能完成动态内存分配和初始化工作的运算符new,以及一个能完成清理与释放内存工作的运算符delete。注意new/delete不是库函数。
我们不要企图用malloc/free来完成动态对象的内存管理,应该用new/delete。由于内部数据类型的“对象”没有构造与析构的过程,对它们而言malloc/free和new/delete是等价的。
3)既然new/delete的功能完全覆盖了malloc/free,为什么C++不把malloc/free淘汰出局呢?这是因为C++程序经常要调用C函数,而C程序只能用malloc/free管理动态内存。
如果用free释放“new创建的动态对象”,那么该对象因无法执行析构函数而可能导致程序出错。如果用delete释放“malloc申请的动态内存”,结果也会导致程序出错,但是该程序的可读性很差。所以new/delete必须配对使用,malloc/free也一样。
相关文章
- 深入理解 HashMap
- 深入理解Oracle数据库主目录结构(oracle数据库主目录)
- 深入解析Oracle数据库客户端(oracle数据库客户端)
- 深入解析MySQL权限授予方式(mysql权限授权)
- 深入体验:从远程连接Linux桌面(连接linux服务器桌面)
- 占用标题:深入了解Linux机器内存占用(查看linux机器内存)
- Linux文件系统卸载实践:深入解析(linux文件系统卸载)
- 深入解析Linux系统下LLVM的功能(linuxllvm)
- 深入解析Oracle触发器的类型区别(oracle触发器类型)
- 深入解析MSSQL实例:掌握关键技巧提高数据库管理效率(mssql实例)
- 深入解析MySQL并发优化技巧,让你的数据库性能大幅提升!(mysql并发优化)
- 深入探究Oracle表子分区:优化性能与提高可靠性(oracle表子分区)
- 机制Java深入解析MySQL锁机制(javamysql锁)
- 深入解析Oracle的版本号码(oracle版本号)
- 深入剖析:Linux系统关键指标解析(Linux系统分析指标)
- 日志深入解析:查看MSSQL运行日志(查看mssql运行)
- 深入了解MSSQL去掉特殊字符的方法(去掉特殊字符 mssql)
- 深入解析Oracle STK技术:高效管理存储空间(oraclestk)
- 深入解析MySQL中不同时间类型的应用与区别(mysql时间的类型)
- 深入探究:SQL Server的小数位数除零处理(sqlserver去0)
- 深入解析Oracle序列号的作用和使用方法(oracle的序列号)
- oerience给领导造福:深入提升SQL Server使用经验(sqlserver ex)
- 深入剖析MySQL中神奇的pi函数(mysql中pi函数)
- 深入解析Oracle 为何无法使用(oracle为什么不能用)
- ubuntu12.10上android编译环境搭建的深入解析
- 深入解析fsockopen与pfsockopen的区别
- PHP线程安全与非线程安全版本的区别深入解析
- 深入解析C语言中常数的数据类型