`
kmplayer
  • 浏览: 498705 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

释放对象数组:delete与delete[]

阅读更多
<<c++ primer>>练习   14.11   中提到:
Account   *parray=new   Account[100];
delete   parray;
delete   []   parray;
方括号的存在会使编译器获取数组大小(size)然后析构函数再被依次应用在每个元素上,一共size次。否则,只有一个元素被析构。
无论哪种情况,分配的全部空间被返还给自由存储区。

我的问题是:为什么无论哪种情况,分配的全部空间被返还给自由存储区?
对于delete parray,为什么不是删除单个Account元素,而是删除了100个.
编译器怎么知道parray这个指针实际指向的是数组还是单个元素,即便知道指向的是自由存储区的数组,这个数组的大小又怎么知道。
难道是编译器辅助行为?

总结:空间释放(肯定有个log记录分配的大小)和调用析构函数(类型识别,不同的编译器实现不同)采用不同的机制.

(1)一般在分配时分配器会自动写一个日志(一般在分配使用得内存之前又一个结构),用于记录分配的大小,分配内容的sizeof等等。
直观得想想,delete和delete[]都是传入一个void*如果不保存日志就无法知道分配时到底是分配了一个还是多个单元.
所以虽然delete和delete[]不同但是分配器在执行释放过程中都会读取这个日志,从而了解到底应该释放多少内存,但是从程序员的角度来说,既然分配了数组,就应该用delete[]

(2)在VC下用汇编跟过delete[]的执行情况,发现这个 "日志 "就是一个4字节长的整数记录数组元素个数,紧挨在数组第一个元素之前.  
但是有个前提:对象类型(或其基类)有显式析构函数.换句话说,析构函数是非trivial的.  
否则的话,数组前面是没有这个日志的.其实对于析构函数是trivial的情况,delete[]时无需调用其析构函数,因此此时VC把delete[]当做delete同样处理.

(3)因为释放数组空间和为数组调用析构函数是两个独立的部分,可以使用不同的机制来实现。
释放空间的机制是需要绝对保证的。因此,即使你不写delete[],它也会将所有空间释放,其机制可以是前置的长度信息,也可以不是(如后置的特征分割符等等)。
而调用析构函数可以一般采用前置长度信息的方式(当然也可以有其他方式)。在没有[]提示时,编译器在调用析构就将它当一个元素,而不会使用数组方式来调用每一个析构函数了。
LS:“但是有个前提:对象类型(或其基类)有显式析构函数.换句话说,析构函数是非trivial的. 否则的话,数组前面是没有这个日志的.”
——这说明,LS使用的编译器在释放数组空间时,并没有用前置的长度信息的方式。由此可见,释放数组空间和为数组调用析构函数确实可以使用不同的机制

(4)delete parray,编译器得到类型信息是Account单个的指针,那么释放时,只调用一次析构函数。
delete[] parray,编译器得到的类型信息是Account[]类型,则按照Account数组来处理,依次调用每个元素的析构函数。

注意,以上是在编译期间就确定下来的,编译器识别到类型信息的不同会决定调用析构函数的情况有不同。

但是对于内存释放,delete操作则不是通过类型信息来确定分配的内存大小,那么内存大小的信息从什么地方得到呢?
当我们使用operator new为一个自定义类型对象分配内存时,实际上我们得到的内存要比实际对象的内存大一些,这些内存除了要存储对象数据外,还需要记录这片内存的大小,此方法称为   cookie。这一点上的实现依据不同的编译器不同。(例如   MFC   选择在所分配内存的头部存储对象实际数据,而后面的部分存储边界标志和内存大小信息。g++   则采用在所分配内存的头4个自己存储相关信息,而后面的内存存储对象实际数据。)当我们使用   delete   operator   进行内存释放操作时,delete   operator   就可以根据这些信息正确的释放指针所指向的内存块。
对于parray指针,可以根据这样的cookie信息来得到指向内存空间的大小,delete parray和delete[] parray都是一样的,同样一个指针,cookie信息是相同的,所以对应的内存都会被释放掉。但是由于编译器理解两种情况下的类型是不同的,所以调用析构函数会有不同。

(5)难道是编译器辅助行为?
没错,就是。不同的编译器可能采用的具体方法有可能不一样,但不管采用什么方法,编译器必须记住那块大小。

(6)转自<<effective c++>>
条款5:对应的new和delete要采用相同的形式

下面的语句有什么错?
string *stringarray = new string[100];
delete stringarray;
一切好象都井然有序——一个new对应着一个delete——然而却隐藏着很大的错误:程序的运行情况将是不可预测的。至少,stringarray指向的100个string对象中的99个不会被正确地摧毁,因为他们的析构函数永远不会被调用。
用new的时候会发生两件事。首先,内存被分配(通过operator new 函数,详见条款7-10和条款m8),然后,为被分配的内存调用一个或多个构造函数。用delete的时候,也有两件事发生:首先,为将被释放的内存调用一个或多个析构函数,然后,释放内存(通过operator delete 函数,详见条款8和m8)。对于 delete来说会有这样一个重要的问题:内存中有多少个对象要被删除?答案决定了将有多少个析构函数会被调用。
这个问题简单来说就是:要被删除的指针指向的是单个对象呢,还是对象数组?这只有你来告诉delete。如果你在用delete时没用括号,delete就会认为指向的是单个对象,否则,它就会认为指向的是一个数组:
string *stringptr1 = new string;
string *stringptr2 = new string[100];
...
delete stringptr1;// 删除一个对象
delete [] stringptr2;// 删除对象数组
如果你在stringptr1前加了"[]"会怎样呢?答案是:那将是不可预测的;
如果你没在stringptr2前没加上"[]"又会怎样呢?答案也是:不可预测。
int这样的固定类型来说,结果也是不可预测的,即使这样的类型没有析构函数。所以,解决这类问题的规则很简单:如果你调用new时用了[],调用delete时也要用[]。如果调用new时没有用[],那调用delete时也不要用[]。

#include <iostream>

using namespace std;

struct foo
{
    ~foo(){}; //去掉后,就不会记录个数了.
};


int main(int argc,char *argv[])
{

     foo* f = new foo;
     delete f;
     f=0;
     delete f;

     foo* fa = new foo[8];
     printf("%u\n", *((char *)fa - 4)); //输出8:辅助析构函数.
     /*
     对于有显式析构函数的对象的数组,
     编译器会在数组前分配4个字节储存数组元素的个数
     (也就是需要调用析构函数的次数),
     因为必须知道数组实际的元素合个数,
     delete[]才能知道需要调用几次析构函数。
     */
     delete fa; //有的编译器这里会有异常.
     //用delete而不是delete[]释放用new[]分配的空间这种行为是undefined的,
     //也就是由编译器实现所决定的
     fa = 0;
     delete fa;

     return 0;
}
分享到:
评论

相关推荐

    C++ new/delete相关知识点详细解析

    一、new和delete创建和释放动态数组:数组类型的变量有三个重要的限制:数组长度固定,在编译时必须知道其长度,数组只在定义它的语句内存在。动态数组:长度固定,编译时不必知道其长度,通常是运行时确定;一直...

    MemoryManager:我正在编写一个内存管理器,以简化C ++的工作。 目标是创建一些东西来替代new和delete并保持速度,同时还要处理垃圾回收并防止内存碎片

    统计记录:是丢失的内存跟踪:是构造函数:是的析构函数:是处理指向基元的指针:是处理指向对象的指针:是处理指向对象静态数组的指针:是处理指向动态数组的指针:是在没有外部引用的情况下检测并释放任意图:...

    35_内存泄漏是什么1

    2.系统资源泄露(Resource Leak) 3.没有将基类的析构函数定义为虚函数 4.在释放对象数组时没有使delete[]是使了delete 5.缺少拷构

    C++11内存管理和多线程编程

     2、每一个New都有一个与之对应的delete进行释放。  3、new 和malloc的区别在于new不但分配了内存还同时创建对象,而malloc只负责分配内存。  4、直接声明数组和声明的new int[5];区别只在于他们存在的位置。...

    C++对象的动态建立与释放详解

    =============下面先给出一个new和delete基本应用的例子,回顾一下它的基本用法============ 代码如下:#include&lt;iostream&gt;using ...//开辟一个存放字符数组(包括10个元素)的空间,返回首元素的地址 int i; for(i=0

    C++智能指针的原理和实现.pdf

    delete:指向⼀个动态独享的指针,销毁对象,并释放与之关联的内存。 使⽤堆内存是⾮常频繁的操作,容易造成堆内存泄露、⼆次释放等问题,为了更加容易和更加安全的使⽤动态内存,C++11中引⼊了智 能指针的概念,⽅...

    C++智能指针.pdf

    如:shared_ptr&lt;A&gt; p3(new A[10], default_delete[]&gt;()),这样就知道我们使⽤智能指针指向了⼀个对象数组,这样就可以正确释放了。 其实,使⽤ shared_ptr 指向对象数组不需要通过删除器的⽅式,只需要在定义 shared...

    零起点学通C++多媒体范例教学代码

    14.10 在对象数组中初始化成员变量 14.11 指针数组 14.12 枚举常量与数组 14.13 多维数组 14.14 多维数组的初始化 14.15 字符数组 14.16 重载数组下标操作符 14.17 总结 第15章 链表 15.1 声明链表结构 15.2 简单的...

    零起点学通C++学习_多媒体范例教学代码

    14.10 在对象数组中初始化成员变量 14.11 指针数组 14.12 枚举常量与数组 14.13 多维数组 14.14 多维数组的初始化 14.15 字符数组 14.16 重载数组下标操作符 14.17 总结 第15章 链表 15.1 声明链表结构 ...

    c++ 班级学生学期成绩管理系统 单继承

    本班级类CClass的对象成员数组需要在构造函数中用new动态分配内存空间,在析构函数中用delete释放。在CClass类中设计包括能够求解最高成绩、最低成绩和平均成绩以及通过学号查找并输出某个学生全部信息(例如Seek()...

    谭浩强C语言程序设计,C++程序设计,严蔚敏数据结构,高一凡数据结构算法分析与实现.rar

    10.4.2 使用字符串指针变量与字符数组的区别 158 10.5 函数指针变量 159 10.6 指针型函数 160 10.7 指针数组和指向指针的指针 161 10.7.1 指针数组的概念 161 10.7.2 指向指针的指针 164 10.7.3 main 函数的参数 166...

    谭浩强C语言程序设计,C++程序设计,严蔚敏数据结构,高一凡数据结构算法分析与实现.rar )

    10.4.2 使用字符串指针变量与字符数组的区别 158 10.5 函数指针变量 159 10.6 指针型函数 160 10.7 指针数组和指向指针的指针 161 10.7.1 指针数组的概念 161 10.7.2 指向指针的指针 164 10.7.3 main 函数的参数 166...

    新手学习C++入门资料

    C++中new和delete是对内存分配的运算符,取代了C中的malloc和free。 标准C++中的字符串类取代了C标准C函数库头文件中的字符数组处理函数。 C++中用来做控制态输入输出的iostream类库替代了标准C中的stdio函数库。...

    又一个VC车牌识别源码 并可定位车牌.rar

    又一个VC车牌识别源码 并可定位车牌...全局的app(应用程序对象)注意 手工分配内存的清除 和CDC对象的删除 以释放系统的GDI资源,每一个new操作符都要对应一个delete,虽然已经弄出来了,还是希望大家好好读读源程序。

    C++开发及测试人员复习资料

    内存分配中,堆分配和栈分配有什么差别? 这两种方式是不相同的。...而new是在堆中分配内存的,而且一经分配则永久保留,直到显式的以delete运算符来释放掉。如果不进行delete,则会造成内存泄露。

    c/c++ 学习总结 初学者必备

    程序在运行的时候用malloc或new申请任意多少的内存,程序员自己负责在何时用free或delete释放内存。动态内存的生存期由我们决定,使用非常灵活,但问题也最多。 5,、解释堆和栈的区别? 答: (1)栈区(stack):由编译器...

    后端,面试,c++ ,找工作

    (3)new可以调用对象的构造函数,对应的delete调用相应的析构函数。 (4)malloc仅仅分配内存,free仅仅回收内存,并不执行构造和析构函数 (5)new、delete返回的是某种数据类型指针,malloc、free返回的是void...

    MysqlDLL,C#操作MYSQL数据库

     Update(DataRow[])//通过为 DataSet 中的指定数组中的每个已插入、已更新或已删除的行执行相应的 INSERT、UPDATE 或 DELETE 语句来更新数据库中的值  Update(DataSet)//通过为指定的 DataTable 中的每个已插入、...

    Visual C++开发经验技巧宝典(第1章)

    0013 delete与delete []的差别 6 0014 符号#、##、#@的用法 6 0015 将某个地址转换为指针 6 0016 常用内存分配及释放函数 6 0017 生成小于100的随机数 7 1.2 类型与变量 7 0018 静态变量的定义及应用 ...

Global site tag (gtag.js) - Google Analytics