3.8 对象的动态建立和释放
1 new和delete基本语法
1)在软件开发过程中,常常需要动态地分配和撤销内存空间,例如对动态链表中结点的插入与删除。在C语言中是利用库函数malloc和free来分配和撤销内存空间的。C++提供了较简便而功能较强的运算符new和delete来取代malloc和free函数。
注意: new和delete是运算符,不是函数,因此执行效率高。
2)虽然为了与C语言兼容,C++仍保留malloc和free函数,但建议用户不用malloc和free函数,而用new和delete运算符。new运算符的例子:
new int; //开辟一个存放整数的存储空间,返回一个指向该存储空间的地址(即指针)
new int(100); //开辟一个存放整数的空间,并指定该整数的初值为100,返回一个指向该存储空间的地址
new char[10]; //开辟一个存放字符数组(包括10个元素)的空间,返回首元素的地址
new int[5][4]; //开辟一个存放二维整型数组(大小为5*4)的空间,返回首元素的地址
float *p=new float (3.14159); //开辟一个存放单精度数的空间,并指定该实数的初值为//3.14159,将返回的该空间的地址赋给指针变量p
3)new和delete运算符使用的一般格式为:
用new分配数组空间时不能指定初值。如果由于内存不足等原因而无法正常分配空间,则new会返回一个空指针NULL,用户可以根据该指针的值判断分配空间是否成功。
1) 应用举例
2 类对象的动态建立和释放
使用类名定义的对象都是静态的,在程序运行过程中,对象所占的空间是不能随时释放的。但有时人们希望在需要用到对象时才建立对象,在不需要用该对象时就撤销它,释放它所占的内存空间以供别的数据使用。这样可提高内存空间的利用率。
C++中,可以用new运算符动态建立对象,用delete运算符撤销对象
比如:
Box *pt; //定义一个指向Box类对象的指针变量pt
pt=new Box; //在pt中存放了新建对象的起始地址
在程序中就可以通过pt访问这个新建的对象。如
cout<<pt->height; //输出该对象的height成员
cout<<pt->volume( ); //调用该对象的volume函数,计算并输出体积
C++还允许在执行new时,对新建立的对象进行初始化。如
Box *pt=new Box(12,15,18);
这种写法是把上面两个语句(定义指针变量和用new建立新对象)合并为一个语句,并指定初值。这样更精炼。
新对象中的height,width和length分别获得初值12,15,18。调用对象既可以通过对象名,也可以通过指针。
在执行new运算时,如果内存量不足,无法开辟所需的内存空间,目前大多数C++编译系统都使new返回一个0指针值。只要检测返回值是否为0,就可判断分配内存是否成功。
ANSI C++标准提出,在执行new出现故障时,就“抛出”一个“异常”,用户可根据异常进行有关处理。但C++标准仍然允许在出现new故障时返回0指针值。当前,不同的编译系统对new故障的处理方法是不同的。
在不再需要使用由new建立的对象时,可以用delete运算符予以释放。如
delete pt; //释放pt指向的内存空间
这就撤销了pt指向的对象。此后程序不能再使用该对象。
如果用一个指针变量pt先后指向不同的动态对象,应注意指针变量的当前指向,以免删错了对象。在执行delete运算符时,在释放内存空间之前,自动调用析构函数,完成有关善后清理工作。
3 编程实践
//1 malloc free函数 c关键字
// new delete 操作符号 c++的关键字
//2 new 在堆上分配内存 delete
//分配基础类型 、分配数组类型、分配对象
//3 new和malloc 深入分析
混用测试、异同比较
结论: malloc不会调用类的构造函数
Free不会调用类的析构函数
1 #include <iostream> 2 using namespace std; 3 4 // 1 5 // malloc free c语言的函数 6 // new delete 操作符 c++的语法 7 8 //2 new 基础类型变量 分配数组变量 分配类对象 9 10 //3 11 12 ////分配基础类型 13 void main1301() 14 { 15 // 16 int *p = (int *)malloc(sizeof(int)); 17 *p = 10; 18 free(p); 19 20 int *p2 = new int; //分配基础类型 21 *p2 = 20; 22 free(p2); 23 // 24 int *p3 = new int(30); 25 printf("*p3:%d \n", *p3); 26 delete p3; 27 28 cout<<"hello..."<<endl; 29 system("pause"); 30 return ; 31 } 32 33 //分配数组变量 34 void main1302() 35 { 36 //c语言分配数组 37 int *p = (int *)malloc(sizeof(int) * 10); //int array[10]; 38 p[0] = 1; 39 free(p); 40 41 //c++分配数组 42 int *pArray = new int[10] ; 43 pArray[1] = 2; 44 delete [] pArray; //数组不要把[] 忘记 45 46 char *pArray2 = new char[25] ; //char buf[25] 47 delete [] pArray2; 48 49 50 cout<<"hello..."<<endl; 51 system("pause"); 52 return ; 53 } 54 55 class Test 56 { 57 public: 58 Test(int _a) 59 { 60 a = _a; 61 cout<<"构造函数执行" <<endl; 62 } 63 64 ~Test() 65 { 66 cout<<"析构函数执行" <<endl; 67 } 68 69 protected: 70 private: 71 int a; 72 }; 73 74 //分配对象new delete 75 //相同 和 不同的地方 new能执行类型构造函数 delete操作符 能执行类的析构函数 76 void main() 77 { 78 //c 79 Test *pT1 = (Test *)malloc(sizeof(Test)); 80 free(pT1); 81 82 //c++ 83 Test *pT2 = new Test(10); 84 delete pT2; 85 86 cout<<"hello..."<<endl; 87 system("pause"); 88 }