new和delete表达式可以用来动态创建和释放单个对象,也可以用来动态创建和释放动态数组。

  定义变量时,必须指定其数据类型和名字。而动态创建对象时,只需指定其数据类型,而不必为该对象命名。new表达式返回指向新创建对象的指针,我们通过该指针访问对象:

 int i; //named, uninitizlized int variable
int *pi = new int; //pi points to dynamically allocated unnamed int
 
   这个new表达式在堆内分配创建了一个整形对象,并返回对象的地址,并用该地址初始化指针pi。
   如果new失败,抛出标准异常std::bad_alloc异常。
 
  1. 动态创建对象的初始化
   动态创建的对象可以初始化变量的方式实现初始化:
   

 int i(); //value of i is 1024
int *pi = new int(); //object to which pi is 1024
string s(,''); //value of s is '9999999999'
string *ps = new string(, ''); //*ps is '9999999999'

C++使用值初始化的方式语法规则初始化动态创建的对象。如果提供了初值,new表达式分配到所需要内存后,用给定的初值初始化该内存。

  2. 动态创建对象的默认初始化

    如果不显式初始化。动态创建的对象与在函数内定义的变量初始化方式相同。对于类类型对象,用该类的默认构造函数初始化;而内置类型的对象则无初始化

 string *ps = new string; //initialized to empty string
int *pi = new int; //pi points to uninitialized int

   正如我们几乎总要初始化定义为变量的对象一样,在动态创建对象时,(几乎)总是对它做初始化也是一个好方法。

   同样也可以对动态创建的对象做值初始化,规则是:对于类类型对象,调用该类的默认构造函数;内置类型初始化为0;

 string *ps = new string() //initialized to empty string
int *pi = new int(); //pi points to int value-initialized to 0
cls *ps = new cls(); //pc points to value-initialized object of type cls

   可以看出,对于类类型的对象,无论程序是明确地不初始化还是要求进行初始化,都会自动调用其默认构造函数初始化该对象。而对于内置类型或没有定义默认构造函数的类型,采用不同初始化方式则有明显的差别:

 int *pi = new int; //pi points to uninitialized int
int *pi = new int(); //pi points to an int value-initialized to 0

第一个语句的int型变量没有初始化,而第二个语句的int型变量初始化为0。

   
  3. 撤销动态创建的对象
    动态创建的对象用完后,程序员必须显式地将对象占用的内存返回给堆。C++提供了delete表达式释放指针所指向的地址空间
    delete pi; //该命令释放pi指向的int型对象所占用的空间,指针本身仍然存在,只是内存空间被回收。
    
    如果指针指向的不是new分配的内存地址,则该指针上使用delete是不合法的。例如,
    int i ;
    int *pi = &i;
      delete pi; 
   上述语句是错误的,i是分配在栈上,栈上的内存不需也不能调用delete删除。
  4. 零值指针的删除
    如果指针的值为0,则在其上做delete操作是合法的,但这样做没有任何意义:
 int *ip = ;
delete pi; //ok: always ok to delete a pointer that is equal to 0

  5. 在delete之后,重设指针的值

    执行语句 delete p; 后,p变成没有定义。在很多机器上,尽管p没有定义, 但仍然存放了它之前所指向对象的地址,然而p所指向的内存已经释放,因此p不再有效。

    删除指针后,该指针编程悬垂指针。悬垂指针指向曾经存放对象的内存,但该对象已经不存在了。悬垂指针往往导致程序错误,而且很检测出来。

    一旦删除了指针指向的对象,立即将指针置为0,这样就非常清楚地表明指针不再指向任何对象。

  6.动态内存的管理、

  (1)删除指向动态分配内存的指针失败,因为无法将该块内存返回给自由存储区,删除动态分配内存失败称为“内存泄露”。内存泄露很难发现,一般需要等应用程序运行一段时间后,耗尽了所有内存空间时,内存泄露才会显露出来。

  (2)读写已删除的对象,如果删除指针所指向的对象之后,将指针置为0,则比较容易检测出这类错误。、

  (3)对同一内存空间使用两次delete表达式。当两个指针指向同一个动态创建的对象,删除就会发生错误,如果在其中一个指针上做delete运算,则该对象的内存空间返回给自由存储区,然后接着delete第二个指针,此时则自由存储区可能被破坏。

  7. new和delete的底层原理

       当你使用new时,有两件事情发生。第一,内存被分配出来(通过operator new的函数)。第二,针对此内存会有一个(或更多)构造函数被调用。

    当你使用delete时,也有两件事情发生:针对此内存会有一个(或更多)的析构函数被调用,然后内存才能被释放(通过名为:operator delete的函数)。

    
创建动态数组:
  数组类型的变量有三个变量的限制:数组长度固定不变,在编译时必须知道其长度,数组在定义它的块语句内存在。  
  1. 动态数组的定义
      new表达式需要指定指针类型以及在方括号中给出的数组维数,该维数可以是任意的复杂表达式。创建数组后,new 将返回指向数组第一个元素的指针。在堆中创建的数组对象时没有名字的,程序员只能通过其间接地访问堆中的对象。
     int *pia = new int[10];、
 
   2. 初始化动态分配的数组
    动态分配时,如果数组元素具有类类型,将使用该类型的默认构造函数实现初始化;如果数组元素是内置类型,则无初始化。也可使用跟在数组长度后面的一对空括号,对数组元素做值初始化:
    int *pia2 = new int[10]();  //array of 10 uninitialized ints
    圆括号要求编译器对数组做值初始化,在本例中即把数组元素都设置为0。
    
    对于动态分配的数组,其元素只能初始化为元素类型的默认值,而不能像数组变量一样,用初始化列表为数组元素提供各不相同的初值。

    
   3. 动态空间的释放

    动态分配的空间必须释放,否则,内存最终将会被耗尽。如果不再需要使用动态创建的数组,程序员必须显式地将其占用的存储空间返回给程序的堆区。C++语言为指针提供了delete[] 表达式释放指针所指向的数组空间:

    delete []pia;

    该语句回收了pia所指向的数组,把相应的内存返回给堆。在关键字delete和指针之间的方括号是必不可少的;它告诉编译器该指针指向的是堆区的数组,而并非是单个对象。

    理论上,回收数组时缺少方括号对,至少会导致运行时少释放了内存空间,从而产生内存泄露。对于某些系统或元素类型,有可能会带来更严重的运行时错误。因此,在释放动态内存时千万不能忘了方括号对

    更多解释详见《Effective C++》条款16。

  malloc/free和new/delete的区别:

  1) malloc和free是C/C++语言的标准库函数,new/delete是C++的运算符。

  2) malloc/free需要库文件支持,new/delete不需要。

  3) new自动计算需要分配空间的大小,malloc需要手工计算字节数

  4)new是类型安全的,而malloc不是,比如:

int* p = new float[]; //编译时指出错误
int* p = (int*)malloc(*sizeof(double)); //编译时无法指出错误

  5)new调用operator new分配足够的空间,并调用相关对象的构造函数,而malloc不能调用构造函数;delete将调用该实例的析构函数,然后调用类的operator delete,以释放该实例占用的空间,而free不能调用析构函数。

【C++】new和delete表达式与内存管理的更多相关文章

  1. 引用计数的cocos2dx对象内存管理和直接new/delete box2d对象内存管理冲突的解决方法

    转载请注明: http://blog.csdn.net/herm_lib/article/details/9316601 项目中用到了cocos2dx和box2d,cocos2dx的内存是基于引用计数 ...

  2. 内存管理运算符new delete与内存管理函数malloc free的区别——已经他们对对象创建的过程。

    (1)内存管理函数与内存管理运算符的区别 内存管理函数有内存分配函数,malloc calloc realloc 以及内存释放函数free. 内存管理运算符有new 和delete. 两种内存管理方式 ...

  3. Cocos2d-x内存管理研究<二>

    http://hi.baidu.com/tzkt623/item/46a26805adf7e938a3332a04   上一篇我们讲了内核是如何将指针加入管理类进行管理.这次我将分析一下内核是如何自动 ...

  4. 【转】C内存管理

    在任何程序设计环境及语言中,内存管理都十分重要.在目前的计算机系统或嵌入式系统中,内存资源仍然是有限的.因此在程序设计中,有效地管理内存资源是程序员首先考虑的问题. 第1节主要介绍内存管理基本概念,重 ...

  5. C++ Primer : 第十二章 : 动态内存之动态内存管理(new和delete)

    C++语言定义了两个运算符来分配和释放动态内存:运算符new分配内存,运算符delete释放new分配的内存. 运算符new和delete 使用new动态分配和初始化对象 在自由空间分配的内存是无名的 ...

  6. 深入理解C++ new/delete, new []/delete[]动态内存管理

    在C语言中,我们写程序时,总是会有动态开辟内存的需求,每到这个时候我们就会想到用malloc/free 去从堆里面动态申请出来一段内存给我们用.但对这一块申请出来的内存,往往还需要我们对它进行稍许的“ ...

  7. C++内存管理(new operator/operator new/operator delete/placement new)

    new operator 我们平时使用的new是new操作符(new operator),就像sizeof一样是语言内置的,不能改变它的含义,功能也是一样的 比如: string *ps = new ...

  8. 深入理解C++中的new/delete和malloc/free动态内存管理

    malloc/free和new/delete的区别 malloc/free是C/C++标准库的函数:new/delete是C++操作符. malloc/free只是动态分配内存空间/释放空间:new/ ...

  9. C++—动态内存管理之深入探究new和delete

    C++中程序存储空间除栈空间和静态区外,每个程序还拥有一个内存池,这部分内存被称为自由空间(free store)或堆(heap).程序用堆来存储动态分配的对象,即,那些程序运行时分配的对象.动态对象 ...

随机推荐

  1. 20155203 实验四《 Android程序设计》实验报告

    一.实验内容 实验项目一:Android Stuidio的安装测试: 参考<Java和Android开发学习指南(第二版)(EPUBIT,Java for Android 2nd)>第二十 ...

  2. 预备作业03: 初体验---虚拟机virtual box

    学习基于VirtualBox虚拟机安装Ubuntu图文教程在自己笔记本上安装Linux操作系统 昨天从下午到晚上一直在研究怎么安装也问了很多同学,也自己查了查百度,经过一晚上终于安装成功,于是今天一早 ...

  3. c++静态变量

    静态变量 一.静态变量 static关键字 static int i; 二.静态变量的运用 .计算函数被调用次数 .返回指针 第45课中有这么一段 int* square3(int *x) { int ...

  4. Ceph学习之路(一)之ceph初识

    一.元数据和元数据管理 (1)元数据 在学习Ceph之前,需要了解元数据的概念.元数据又称为中介数据.中继数据,为描述数据的数据.主要描述数据属性的信息,用来支持如指示存储位置.历史数据.资源查找.文 ...

  5. 开源工具 | 手游自动化框架GAutomator,新增iOS系统和UE4引擎支

    WeTest 导读 GAutomator是腾讯WeTest推出的手游自动化测试框架,已用于腾讯多个手游项目组的自动化测试. 1.GAutomator诞生背后 研究过手游自动化测试的同学都知道,虽然市场 ...

  6. kobject和kset的一些学习心得

    #include <linux/module.h> #include <linux/kernel.h> #include <linux/kobject.h> #in ...

  7. 学习HTML 第一节.小试牛刀

    此贴并非教学,主要是自学笔记,所述内容只是些许个人学习心得的记录和备查积累,难以保证观点正确,也不一定能坚持完成. 如不幸到访,可能耽误您的时间,也难及时回复,贴主先此致歉.如偶有所得,相逢有缘,幸甚 ...

  8. 韦大仙--LoadRunner压力测试:详细操作流程

    一. 录制脚本 1.安装完毕后,创建脚本: 点击OK之后,会弹出网址,之后创建Action,每进一个页面添加一个Action,录制结束后,终止录制. 二. 修改脚本 1.脚本参数化 将登录的用户名密码 ...

  9. [转]50 Tips for Working with Unity (Best Practices)

    About these tips These tips are not all applicable to every project. They are based on my experience ...

  10. ObjectAnimator实现菜单的弹出(扇形)

    用ObjectAnimator 实现菜单的弹出 首先是菜单的图片资源和布局 布局中使用FrameLaout 将菜单唤出对应的imageView放在布局的最后面来隐藏菜单详细内容. <?xml v ...