windows 内存分配方式
windows 内存分配方式
2011年02月25日
1.Win32的堆分配函数
每一个进程都可以使用堆分配函数创建一个私有的堆──调用进程地址空间的一个或者多个页面。DLL创建的私有堆必定在调用DLL的进程的地址空间内,只能被调用进程访问。
HeapCreate用来创建堆;HeapAlloc用来从堆中分配一定数量的空间,HeapAlloc分配的内存是不能移动的;HeapSize可以确定从堆中分配的空间的大小;HeapFree用来释放从堆中分配的空间;HeapDestroy销毁创建的堆。
2.Windows传统的全局或者局部内存分配函数
由于Win32采用平面内存结构模式,Win32下的全局和局部内存函数除了名字不同外,其他完全相同。任一函数都可以用来分配任意大小的内存(仅仅受可用物理内存的限制)。用法可以和Win16下基本一样。
Win32下保留这类函数保证了和Win16的兼容,所以不建议再使用了。
3.C语言的标准内存分配函数
C语言的标准内存分配函数包括以下函数:。
malloc,calloc,realloc,free,等。
这些函数最后都映射成堆API函数,所以,malloc分配的内存是不能移动的。这些函数的调式版本为malloc_dbg,calloc_dbg,realloc_dbg,free_dbg,等。
4.Win32的虚拟内存分配函数
虚拟内存API是其他API的基础。new->malloc->HeapAlloc->VirtualAlloc->驱动程序的_PageAlloc。
虚拟内存API以页为最小分配单位,X86上页长度为4KB,可以用GetSystemInfo函数提取页长度。
虚拟内存分配函数包括以下函数:
LPVOID VirtualAlloc(
LPVOID lpvAddress,
DWORD cbSize,
DWORD fdwAllocationType,
DWORD fdwProtect);
该函数用来分配一定范围的虚拟页。
参数1指定起始地址;
参数2指定分配内存的长度;
参数3指定分配方式,取值MEM_COMMINT或者MEM_RESERVE;
参数4指定控制访问本次分配的内存的标识,取值为PAGE_READONLY、 PAGE_READWRITE或者PAGE_NOACCESS。
LPVOID VirtualAllocEx(
HANDLE process,
LPVOID lpvAddress,
DWORD cbSize,
DWORD fdwAllocationType,
DWORD fdwProtect);
该函数功能类似于VirtualAlloc,但是允许指定进程process。VirtaulFree、VirtualProtect、VirtualQuery都有对应的扩展函数。
BOOL VirtualFree(
LPVOID lpvAddress,
DWORD dwSize,
DWORD dwFreeType);
该函数用来回收或者释放分配的虚拟内存。
参数1指定希望回收或者释放内存的基地址;
如果是回收,参数2可以指向虚 拟地址范围内的任何地方,如果是释放,参数2必须是VirtualAlloc返回的地址;
参数3指定是否释放或者回收内存,取值为 MEM_DECOMMINT或者MEM_RELEASE。
BOOL VirtualProtect(
LPVOID lpvAddress,
DWORD cbSize,
DWORD fdwNewProtect,
PDWORD pfdwOldProtect);
该函数用来把已经分配的页改变成保护页。
参数1指定分配页的基地址;
参数2指定保护页的长度;
参数3指定页的保护属性,取值PAGE_READ、PAGE_WRITE、PAGE_READWRITE等等;
参数4用来返回原来的保护属性。
DWORD VirtualQuery(
LPCVOID lpAddress,
PMEMORY_BASIC_INFORMATION lpBuffer,
DWORD dwLength
);
该函数用来查询内存中指定页的特性。
参数1指向希望查询的虚拟地址;
参数2是指向内存基本信息结构的指针;
参数3指定查询的长度。
BOOL VirtualLock(
LPVOID lpAddress,
DWORD dwSize);
该函数用来锁定内存,锁定的内存页不能交换到页文件。
参数1指定要锁定内存的起始地址;
参数2指定锁定的长度。
BOOL VirtualUnLock(
LPVOID lpAddress,
DWORD dwSize);
参数1指定要解锁的内存的起始地址;
参数2指定要解锁的内存的长度。
转自:http://blog.csdn.net/jifengszf/archive/2007/11/14/ 1884743.aspx
详细调用关系:
malloc(msvcrt.malloc) => kernel32.HeapAlloc(ntdll.RtlAllocateHeap)
new( msvcrt.??2@YAPAXI@Z) => kernel32.HeapAlloc(ntdll.RtlAllocateHeap)
kernel32.LocalAlloc => ntdll.RtlAllocateHeap
kernel32.GlobleAlloc => ntdll.RtlAllocateHeap
kernel32.HeapAlloc == ntdll.RtlAllocateHeap
kernel32.VirtualAlloc => kernel32.VirtualAllocEx
kernel32.VirtualAllocEx => ntdll.NtAllocateVirtualMemory
ntdll.RtlAllocateHeap => ntdll.NtAllocateVirtualMemory
ntdll.NtAllocateVirtualMemory => &nb