日期:2014-05-17  浏览次数:20765 次

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