delphi.memory.分配及释放---New/Dispose, GetMem/FreeMem及其它函数的区别与相同,内存分配函数

来自:http://www.cnblogs.com/qiusl/p/4028437.html?utm_source=tuicool&utm_medium=referral

--------------------------------------------------------------------------------------------------------

我估摸着内存分配+释放是个基础函数,有些人可能没注意此类函数或细究,但我觉得还是弄明白的好。

介绍下面内存函数前,先说一下MM的一些过程,如不关心可忽略:

TMemoryManager = record
  GetMem: function(Size: Integer): Pointer;
  FreeMem: function(P: Pointer): Integer;
  ReallocMem: function(P: Pointer; Size: Integer): Pointer;
end;

var
  MemoryManager: TMemoryManager = (
    GetMem: SysGetMem;
    FreeMem: SysFreeMem;
    ReallocMem: SysReallocMem);

以上是D7版本的MM函数,其中变量MemoryManager我称为MM函数,请注意。

D2005-D2007以上版本(不确认哪个版本),MM函数多了AllocMem及RegisterLeak/UnRegisterLeak函数,与本文无关,就不多说了。

第三方MM接管的就是这MM的几个函数,达到外挂目地,而Sys打头的SysGetMem, SysFreeMem, SysReallocMem则为本身系统自带的MM处理。

一:New/Dispose

此两函数,估计学delphi/pascal,就知道:为record/object此类数据进行分配和释放内存块

然后分配与释放是调用的是GetMem/FreeMem函数,与GetMem/FreeMem不同之处是:

New()在GetMem后,进行了initialize(x)操作,即对record/object的数据进行初始化的操作.

initialize函数,在system单元,该函数说白了,即对record/object里面中,含有string,interface, dync array,variant,record,array的字段,进行初始化为0(清空).

这一步很重要,因为GetMem返回的内存块可能重复使用过的,使用过的,表示有值。

有值的情况下,再重新赋值,就表示旧地址对应的数据要先清空,清空随机地址的数据?AV就会出现了...

(不要想着,在GetMem后,进行每字段初始化,容易出错的就是这个,在有以上以字段的情况下,如果需要手动初始化,必须用fillchar,原因如上。)

与之相反的Dispose()亦同,反操作,进行清空:finalize(x)后,再进行FreeMem,以保证record/object中,

string,interface/dyncarray字段,不会因为直接调用FreeMem而泄露(leak)

总结是:

a: New==> GetMem(p, sizeof(TDataType)) + Initialize(p^) ==>AllocMem(sizeof(TDataType));

它与AllocMem区别是:initialize(x)不会对每个字节清0,只针对于某些字段清0.

Dispose == Finalize(p^) + FreeMem(p);

没有可代替的函数,也不能少finalize(p^)这步操作,否则会有leak.

b: record/object的指针类型,最好使用此对函数进行分配及释放。当然你也可以去自维护record/object里面的字段生存期。

c: 如果调用system.Initialize/Finalize,出现提示:

[Hint] Unit1.pas(43): Expression needs no Initialize/Finalize

表示record/object里面的字段,没有包含string,interface,dync array,variant,record,array

即表示不需要调用Initialize或Finalize进行操作。

d: 多说一句:每个warn/hint都有其作用,请勿忽略,说不定小BUG就在其中,请关注它们或干掉它们。

二:GetMem/FreeMem

GetMem/FreeMem是MM的分配与释放内存块函数,多说一些是与之相关的:此两函数,会因为分配或释放失败而抛出异常(exception)

而MM对应的标准分配与释放函数是以返回值形态进行处理的,即失败了,只会返回空值(nil)或非0,而不是异常。

也就是说Get/FreeMem是针对于MM的标准函数进行了异常封装。

异常信息:

GetMem fail => Out of memory.

分配失败,一般只会是进程的可用内存分配完毕,通常在内存泄露的情况下才会发生。

FreeMem fail => Invalid pointer operation

两次FreeMem同地址,第二次就有这invalid pointer异常了。:)

三:GetMemory/FreeMemory

Get/FreeMemory与GetMem/FreeMem基本相同,唯一不相同的是,它直接以MM的对应函数的返回值作为返回,而不进行异常处理。

即:GetMem调用MM.GetMem返回为nil,则有异常,而GetMemory则直接返回nil,交给调用者处理

FreeMem调用MM.FreeMem返回非0(错误释放),则异常,而FreeMemory则直接返回0或非0,给调用者处理。

这点非常有用,在写程序时,可以减少异常,或者在Get/Free出现错误时,写句assert(...),让程序中断下来,检查并调试。

四:SysGetMem/SysFreeMem

SysGetMem/SysFreeMem与GetMemory/FreeMemory基本相同,区别在于,它直接调用MM的实现函数,

则不是经过MM的管理器指针再行跳转。

即说:SysGet/SysFreeMem,它使用的是系统自带的MM分配释放函数,当第三方MM加入后,以上三对函数,

都会由第三方MM接管,但SysGet/SysFreeMem它还是调用的本系统自带的MM函数处理,与第三方MM无关。

五:其它

其它还有些Delphi单元的分配释放函数,不过基本是从以上四对函数扩展出来,就不说明了

当然也有从API扩展出来的分配+释放函数,则不在此列,它与D系统的MM扩展无关。

总结:

New+Dispose与GetMem+FreeMem,是基于VCL异常机制保护的分配+释放函数。

GetMemory+FreeMemory与SysGetMem+SysFreeMem是由调用者自行控制返回,来决定是否返回异常或错误处理。

完。

2014.10.19 by qsl

时间: 2024-10-28 11:00:20

delphi.memory.分配及释放---New/Dispose, GetMem/FreeMem及其它函数的区别与相同,内存分配函数的相关文章

delphi.memory.分配及释放---New/Dispose, GetMem/FreeMem及其它函数的区别与相同

来自:http://www.cnblogs.com/qiusl/p/4028437.html ------------------------------------------------------------------ 我估摸着内存分配+释放是个基础函数,有些人可能没注意此类函数或细究,但我觉得还是弄明白的好. 介绍下面内存函数前,先说一下MM的一些过程,如不关心可忽略: TMemoryManager = record GetMem: function(Size: Integer): P

Delphi的分配及释放---New/Dispose, GetMem/FreeMem及其它函数的区别与相同

转载自:http://www.cnblogs.com/qiusl/p/4028437.html?utm_source=tuicool 我估摸者内存分配+释放是个基础的函数,有些人可能没注意此类函数或细究,但我觉得还是弄明白好. 介绍下面内存函数之前,先说一下MM的一些过程,如不关心课忽略: TMemoryManger = record GetMem: function(Size: Integer): Pointer; FreeMem: function(P: Pointer): Integer;

Java中String、StringBuffer和StringBuilder的区别和堆栈内存分配

Java中的String类是一个很常用,但最不注意其细节的类,因此大多数面试会那这个类做文章.比如String str = new String("hello");开辟了几个内存空间,String和StringBuffer的区别等等.下面就做一个我的理解: String是一个被final修饰的类,它是不能被继承的.StringBuffer也是被final修饰的类. 一.堆内存和栈内存 在JVM中,堆内存是内存空间存放的是对象实例化的内容(程序的数据),栈内存存放的是对象的名称,其内容是

终于懂了:Delphi的函数名不是地址,取地址必须遵守Object Pascal的语法(Delphi和C的类比:指针、字符串、函数指针、内存分配等)good

这点是与C语言不一样的地方,以前我一直都没有明白这一点,所以总是不明白:函数地址再取地址算怎么回事? ---------------------------------------------------------------------------------------------------------------- 在学习Delphi的时候,一个很好的建议是和C/C++去类比着学习,从指针,到内存管理,到数组,到面向对象……各个方面,都是有很多可以相似和或者也有不同的方,类比着学习,一

Com组件的内存分配和释放,CredentialProvider SHStrDup 字符串拷贝问题

一.简介 熟悉CredentialProvider的同学应该知道,他为一个Com组件,于是,在这里的内存分配(字符串拷贝)的一系列操作就要按照con的标准来. 二.Com组件的内存分配和释放 COM提供的任务内存分配器,通过IMalloc接口调用.此接口由CoGetMalloc返回.分配内存可以使用IMalloc::Alloc,释放可由IMalloc::Free完成. 为了简化COM库封装了3个API函数来用于内存的分配和释放,如下: void *CoTaskMemAlloc(ULONG cb)

内存分配方式和调试机制

内存分配方式和调试机制 M内存分配 内存分配函数 MFCWin32或者C语言的内存分配API,有四种内存分配API可供使用. Win32的堆分配函数 每一个进程都可以使用堆分配函数创建一个私有的堆──调用进程地址空间的一个或者多个页面.DLL创建的私有堆必定在调用DLL的进程的地址空间内,只能被调用进程访问. HeapCreate用来创建堆:HeapAlloc用来从堆中分配一定数量的空间,HeapAlloc分配的内存是不能移动的:HeapSize可以确定从堆中分配的空间的大小:HeapFree用

.NET的堆和栈04,对托管和非托管资源的垃圾回收以及内存分配

在" .NET的堆和栈01,基本概念.值类型内存分配"中,了解了"堆"和"栈"的基本概念,以及值类型的内存分配.我们知道:当执行一个方法的时候,值类型实例会在"栈"上分配内存,而引用类型实例会在"堆"上分配内存,当方法执行完毕,"栈"上的实例由操作系统自动释放,"堆"上的实例由.NET Framework的GC进行回收. 在" .NET的堆和栈02,值类型和

技术回归01-Windows内存分配工具

很久没有写技术方面的东西了,这半年主要是在学习别人的东西,对自己提高比较大,算是一次技术回笼吧,这次学习之旅目的是结束技术方面的专注,开始向应用方面找突破口,也就是完成技术积累或者为技术的积累做坚实的准备. c/C++的一个让人疯狂的地方就是内存管理,非法访问.越界.野指针.泄漏.内存分配器等诸多问题,有时候一个编程老手也会迷惘困惑.Crt有一些堆栈检查的函数可以完成基本的内存状况检查,MFC也有一些简单的对象检查机制,当然好的算是java..net等sdk的超重量级封装了,即使发生对象错误也能

Linux内核中常见内存分配函数

1.原理说明 Linux内核中采用了一种同时适用于32位和64位系统的内存分页模型,对于32位系统来说,两级页表足够用了,而在x86_64系统中,用到了四级页表,如图2-1所示.四级页表分别为: l   页全局目录(Page Global Directory) l   页上级目录(Page Upper Directory) l   页中间目录(Page Middle Directory) l   页表(Page Table) 页全局目录包含若干页上级目录的地址,页上级目录又依次包含若干页中间目录