总结并剖析malloc/free和new/delete之间关系和差异

来源:互联网 发布:java哪些类不能被继承 编辑:程序博客网 时间:2024/05/20 09:25

总结并剖析malloc/free和new/delete之间关系和差异

两者相同点

1.都可以申请动态堆内存。


两者不同点

1.new/delete是C++的操作符,malloc/free是C/C++的标准库函数。
2.new申请的可以理解为对象,new时会调用构造函数,返回指向该对象的指针,delete时调用析构函数;malloc申请的只是内存,不是对象。

3.new/delete是保留字,不需要头文件支持;malloc/free需要头文件库函数支持。


注意事项

1.用new申请的内存,必须用delete释放。
2.用new[]申请的内存,必须用delete[]释放。
3.delete释放内存后,指针值不变,良好的风格是释放后指针置为NULL,如,delete p; p = NULL。
4.用malloc申请的内存,必须用free释放。


使用

#include "stdafx.h"  #include <stdio.h>  #include "stdlib.h"  #include <string.h>    struct Stu  {      char    name[32];      int     age;  };    int main()  {  /**************************** 基本用法 **********************************/        //申请一个int类型      int *p1 = new int;      //直接申请赋值  int* p1 = new int(3);      int *p2 = (int*)malloc(sizeof(int));        //申请一个char类型      char *p3 = new char;    //直接申请赋值    char *p3 = new char('c');      char *p4 = (char*)malloc(sizeof(char));        //申请一个int型一维数组      int *p5 = new int[5];   //直接申请赋值    int *p5 = new int[5]{1,2,3,4,5};      int *p6 = (int*)malloc(sizeof(int)*5);            //申请一个char型一维数组      char* p7 = new char[6]; //直接申请赋值  char* p7 = new char[3]{'a', 'v', 'c'};      char* p8 = (char*)malloc(sizeof(char)*6);            //申请一个int型二维数组      int(*p9)[2] = new int[2][2]; //直接申请赋值   int(*p9)[2] = new int[2][2]{ 1,2,3,4 };      int(*p10)[2] = (int(*)[2])malloc(sizeof(int)*2*2);            //申请一个char型二维数组      char(*p11)[2] = new char[2][2];      char(*p12)[2] = (char(*)[2])malloc(sizeof(char)*2*2);    /***************************** 申请二级指针内存 **********************************/            //申请二级指针(new,delete)      char** p13 = new char*[2];      p13[0] = "aaaaaaaaaa";      p13[1] = "vvvvvvvvvv";      delete p13;        //申请二级指针(malloc, free)      char** p14 = (char**)malloc(sizeof(char*)*2);      p14[0] = "cccccccc";      p14[1] = "dddddddd";      delete p14;    /****************************** 申请结构体内存 *********************************/        //new delete      Stu* pStu1 = new Stu;      Stu* pStu2 = new Stu{"wpf", 10};      Stu* pStu3 = new Stu[1024];      delete pStu1;      delete pStu2;      delete[] pStu3;        //malloc free      Stu* pStu4 = (Stu*)malloc(sizeof(Stu));      memset(pStu4, 0, sizeof(Stu));      free(pStu4);        getchar();  }  

剖析new/delete、new[]/delete[]到底做了些什么事情

一.基本使用

当我们想要对一段未命名内存分配内容的时候,只能通过指针来访问该内存,我们可以用指针配合new运算符来开辟内存(malloc也可以实现)。我们需要告诉new我们要在内存里存储什么样的数据类型,使用模式如下:

typeName * Pointer_name = new typeName;

在使用new开启了内存空间存储了数据以后,我们在使用完毕后需要使用delete释放掉开辟的内存,方法很简单:

delete Pointer_name;

注意两点:

1.delete只能释放new开启的内存,不要用delete释放声明变量所开辟的内存。

2.delete用来作用于new开辟的地址而不是new对应的指针,例如:

int * Pt = new int;

int * Pg = Pt;

delete Pg;

此处,指针Pg指向new开辟的内存,所以释放掉Pg即可,不需要释放掉new对应的指针Pt。


二.动态数组

如果通过声明来定义数组,程序会在编译时对它分配内存,因此必须明确数组的大小。而如果我们通过使用new来创建动态数组,则是在程序运行的时候才对它分配内存,且可以在程序运行时选择数组的长度。格式如下:

typeName * Pointer_name = new typeName [num_element];

new运算符返回数组的第一个地址,被分配给指针。

释放动态数组内存的方法也很简单:

delete [] Pointer_name;

这个方括号告诉程序,应该释放整个数组的内存,而不是指针指向元素的内存。


使用new遵循原则:

  a.用new申请的内存,必须用delete释放。

  b.用new[]申请的内存,必须用delete[]释放。

  c.delete释放内存后,指针值不变,良好的风格是释放后指针置为NULL,如,delete p; p = NULL。

使用

1.申请一个对象

int* p1 = new int;  delete p1;  p1 = NULL;
2.申请多个对象
int* p1 = new int[12];  delete[] p1;  p1 = NULL;  
3.申请一个长度为1024的char数组

char* pArray = new char[1024];  for (int i=0; i < 1024; i++)  {      pArray[i] = i;  }  delete[] pArray;  pArray = NULL;
4.申请一个类对象

#include <stdio.h>    class Student  {  public:      char name[32];      int  age;  };  int main()  {      Student* pStu = new Student();      delete pStu;      pStu = NULL;      return 1;  }  
5.申请1024个类对象
#include <stdio.h>    class Student  {  public:      int age;      Student()      {          ...      }      ~Student()      {          ...      }  };    int main()  {      Student* pStu = new Student[1024];      for (int i=0; i<1024; i++)      {          pStu[i].age = i+1;      }      delete[] pStu;      pStu = NULL;      return 1;  }  
new多个对象不能传参数,要求该类必须有默认构造函数。




原创粉丝点击