• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            woaidongmao

            文章均收錄自他人博客,但不喜標(biāo)題前加-[轉(zhuǎn)貼],因其丑陋,見諒!~
            隨筆 - 1469, 文章 - 0, 評論 - 661, 引用 - 0
            數(shù)據(jù)加載中……

            Windows平臺下的內(nèi)存管理

            1. Windows平臺下主要的內(nèi)存管理途徑

            申請

            釋放

            new

            delete

            malloc

            free

            CoTaskMemAlloc

            CoTaskMemFree

            IMalloc::alloc

            IMalloc/free

            GlobalAlloc

            GlobalFree

            LocalAlloc

            LocalFree

            HeapAlloc

            HeapFree

            VirtualAlloc

            VirtualFree


            . 調(diào)用關(guān)系

            clip_image002

            第一層:Win32 API作為系統(tǒng)的接口,提供了一組操作虛擬內(nèi)存的接口;

            第二層:Heap作為虛擬內(nèi)存的一部分,Win32 API又提供了一組操作Heap內(nèi)存的接口,但是這些接口是建立在操作虛擬內(nèi)存的接口的基礎(chǔ)上。

            第三層:Windows平臺下的C Run-Time Library 又利用Heap API來實現(xiàn)mallocfree

            由此我們可以看出,這些動態(tài)內(nèi)存操作方式之間存有單一的層次關(guān)系,位于這個層次的最低層的是Virtual Memory API,可以說這些方式都是建立在Virtual Memory API的基礎(chǔ)上。

            調(diào)用關(guān)系如下表所示為 : new -> malloc -> HeapAlloc -> VirtualAlloc -> 驅(qū)動程序的_PageAlloc

            調(diào)用者

            被調(diào)用者

            msvcrt.malloc

            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

            ntdll.KiFastSystemCall

            ntdll.KiFastSystemCall

            sysenter指令 (0F34)

            3. 方法解析

            3.1 Virtual Memory API

                作為Windows系統(tǒng)提供的最"核心"的對虛擬內(nèi)存操作的接口,也作為其他幾種方式的基礎(chǔ),Virtual Memory API應(yīng)該在幾種方式中是最通用,也是功能最強大的一種方式。在Windows里內(nèi)存管理是分為兩部份,全局內(nèi)存是系統(tǒng)管理的內(nèi)存,因而所有進(jìn)程都可以訪問的內(nèi)存,而每一個進(jìn)程又有自己的內(nèi)存空間,這就是虛擬內(nèi)存空間了,而虛擬內(nèi)存的空間比較大,當(dāng)物理內(nèi)存不足時,系統(tǒng)會把虛擬內(nèi)存的數(shù)據(jù)保存到硬盤里,這樣只要硬盤的空間足夠大,每個進(jìn)程就可以使用3G的內(nèi)存。虛擬內(nèi)存分配可以作為程序里分配內(nèi)存的主要方式,比如大量的數(shù)據(jù)緩沖區(qū),動態(tài)分配內(nèi)存的空間。使用VirtualAlloc函數(shù)來分配內(nèi)存的速度要比全局內(nèi)存要快。

               1:   LPVOID  WINAPI  VirtualAlloc( __in_opt LPVOID lpAddress, __in  SIZE_T dwSize,  __in   DWORD flAllocationType,   __in  DWORD flProtect );

            lpAddress是指定內(nèi)存開始的地址。

            dwSize是分配內(nèi)存的大小。

            flAllocationType是分配內(nèi)存的類型。

            flProtect是訪問這塊分配內(nèi)存的權(quán)限。

               1:  void MemVirtual(void) {

               2:      //分配新內(nèi)存大小。

               3:      UINT nNewSize = (UINT) ceil(1500 / 1024.0) * 1024;

               4:      PBYTE pNewBuffer = (PBYTE) VirtualAlloc(NULL,nNewSize,MEM_COMMIT,PAGE_READWRITE);

               5:      if (pNewBuffer){

               6:         //測試虛擬內(nèi)存。

               7:         ZeroMemory(pNewBuffer,1500);

               8:         memcpy(pNewBuffer,_T("分配虛擬內(nèi)存成功\r\n"),sizeof(_T("分配虛擬內(nèi)存成功\r\n")));

               9:         OutputDebugString((LPWSTR)pNewBuffer);

              10:         //釋放分配的內(nèi)存,第三個參數(shù)一定是MEM_RELEASE

              11:         VirtualFree(pNewBuffer,0,MEM_RELEASE);

              12:      }

              13:  }

            3.2 Heap Memory API

            在進(jìn)程私有的內(nèi)存空間里分配里,有兩種分配情況,一種上基于棧式的內(nèi)存分配,另一種是基于堆內(nèi)存的分配。使用堆內(nèi)存分配是使用HeapAlloc函數(shù)來實現(xiàn)的,也就是實現(xiàn)new操作符分配內(nèi)存時會調(diào)這個函數(shù)。這里的"Heap"指的是進(jìn)程擁有的一種對象(Windows中有很多對象,例如WINDOWICONBRUSH),當(dāng)我們創(chuàng)建一個Heap對象的時候,我們就可以獲得這個對象的Handle,然后我們就可以使用這個handle來使用動態(tài)內(nèi)存,最后銷毀這個對象。

               1:  LPVOID WINAPI HeapAlloc(__in HANDLE hHeap,__in DWORD dwFlags,__in SIZE_T dwBytes);

            hHeap是進(jìn)程堆內(nèi)存開始位置。
            dwFlags
            是分配堆內(nèi)存的標(biāo)志。
            dwBytes
            是分配堆內(nèi)存的大小。

               1:  void MemHeap(void){

               2:      const int nHeapSize = 1024;

               3:      PBYTE pNewHeap = (PBYTE) ::HeapAlloc(GetProcessHeap(), 0, nHeapSize);

               4:      if (pNewHeap){

               5:        //測試分配堆內(nèi)存。

               6:        ZeroMemory(pNewHeap,nHeapSize);

               7:        memcpy(pNewHeap,_T("分配堆內(nèi)存成功\r\n"),sizeof(_T("分配堆內(nèi)存成功\r\n")));

               8:        OutputDebugString((LPWSTR)pNewHeap);

               9:        //釋放內(nèi)存

              10:        BOOL bRes = ::HeapFree(GetProcessHeap(), 0, pNewHeap);

              11:        if (bRes != TRUE){

              12:              OutputDebugString(_T("釋放內(nèi)存出錯\r\n"));

              13:          }

              14:      }

              15:  }

            3.3 LocalAlloc/GlobalAlloc

            這兩個函數(shù)是Win16 API中遺留下來的兩個函數(shù),Win32 API為了保持兼容性才包含了這兩個函數(shù)。這兩個函數(shù)內(nèi)部是通過Heap Memory API來操作一個"特殊"Heap對象:進(jìn)程的默認(rèn)堆對象。每一個進(jìn)程在初始化的時候,都會創(chuàng)建一個默認(rèn)的Heap對象,在進(jìn)程結(jié)束的時候銷毀這個默認(rèn) 的Heap對象。LocalAllocGlobalAlloc的區(qū)別僅表現(xiàn)在Win16環(huán)境下,在Win16環(huán)境下,內(nèi)存的地址是通過段:段內(nèi)偏移量 來獲取的,LocalAlloc()只能在同一段內(nèi)分配內(nèi)存,而GlobalAlloc可以跨越段邊界訪問內(nèi)存。 在Win32環(huán)境下內(nèi)存訪問不存在這樣的限制,所以他們表現(xiàn)出相同的功能。由于Heap Memory API完全可以實現(xiàn)他們兩個的功能,所以在Win32下不推薦使用這兩個函數(shù)。

            Windows系統(tǒng)里,有一項功能非常實用,就是剪貼板功能,它能夠從一個程序里與另一個程序進(jìn)行數(shù)據(jù)交換的功能,也就是說兩個進(jìn)程上是可以共享數(shù)據(jù)。要實現(xiàn)這樣的功能,Windows系統(tǒng)在底層上有相應(yīng)的支持,就是高端地址的內(nèi)存是系統(tǒng)內(nèi)存,這樣就可以不同的進(jìn)程進(jìn)行共享數(shù)據(jù)了。因此,調(diào)用函數(shù)GlobalAlloc來分配系統(tǒng)內(nèi)存,讓不同的進(jìn)程實現(xiàn)共享數(shù)據(jù),也就是剪貼板功能,可以在一個進(jìn)程內(nèi)分配內(nèi)存,在另一個進(jìn)程里訪問數(shù)據(jù)后刪除內(nèi)存。

               1:  HLOCAL WINAPI LocalAlloc(__in UINT uFlags,__in SIZE_T uBytes);

               2:  HGLOBAL WINAPI GlobalAlloc (__in UINT uFlags, __in SIZE_T dwBytes);

            示例代碼:

               1:  void MemGlobal(void) {

               2:      //分配全局內(nèi)存。

               3:      BYTE* pGlobal = (BYTE*)::GlobalAlloc(GMEM_FIXED,1024);

               4:      if (!pGlobal) {

               5:          return;

               6:      } else {

               7:          //測試全局內(nèi)存

               8:          ZeroMemory(pGlobal,1024);

               9:          memcpy(pGlobal,_T("分配內(nèi)存成功\r\n"),sizeof(_T("分配內(nèi)存成功\r\n")));

              10:          OutputDebugString((LPWSTR)pGlobal);

              11:      }

              12:      //釋放全局內(nèi)存。

              13:      ::GlobalFree((HGLOBAL)pGlobal);

              14:  }

            3.4 malloc/free

                 這兩個函數(shù)是使用頻率最高的兩個函數(shù),由于他們是標(biāo)準(zhǔn)C庫中的一部分,所以具有極高的移植性。這里的"移植性"指的是使用他們的代碼可以在不同的平臺下編 譯通過,而不同的平臺下的C Run-Time Library的具體實現(xiàn)是平臺相關(guān)的,在Windows平臺的C Run-Time Library中的malloc()free()是通過調(diào)用Heap Memory API來實現(xiàn)的。值得注意的是C Run-Time Library擁有獨立的Heap對象,我們知道,當(dāng)一個應(yīng)用程序初始化的時候,首先被初始化的是C Run-Time Library,然后才是應(yīng)用程序的入口函數(shù),而Heap對象就是在C Run-Time Library被初始化的時候被創(chuàng)建的。

                  對于動態(tài)鏈接的C Run-Time Library,運行庫只被初始化一次,而對于靜態(tài)連接的運行庫,每鏈接一次就初始化一次,所以對于每個靜態(tài)鏈接的運行庫都擁有彼此不同的Heap 對象。這樣在某種情況下就會出問題,導(dǎo)致程序崩潰,例如一個應(yīng)用程序調(diào)用了多個DLL,除了一個DLL外,其他的DLL,包括應(yīng)用程序本身動態(tài)連接運行庫,這樣他們就使用同一個Heap對象。而有一個DLL使用靜態(tài)連接的運行庫,它就擁有一個和其他DLL不同的Heap 對象,當(dāng)在其他DLL中分配的內(nèi)存在這個DLL中釋放時,問題就出現(xiàn)了。

            3.5 關(guān)鍵詞new/關(guān)鍵詞delete

                 這兩個詞是C++內(nèi)置的關(guān)鍵詞(keyword)。當(dāng)C++編譯器看到關(guān)鍵詞new的時候,例如:

                 CMyObject* pObj = new CMyObject;

                 編譯器會執(zhí)行以下兩個任務(wù):

                a) 在堆上動態(tài)分配必要的內(nèi)存。這個任務(wù)是由編譯器提供的一個全局函數(shù)void* ::operator new(size_t)來完成的。值得注意的是任何一個類都可以重載這個全局函數(shù)。如果類重載了這個函數(shù)的化,被類重載的那個會被調(diào)用。

                b) 調(diào)用CMyObject的構(gòu)造函數(shù)來初始化剛剛生成的對象。當(dāng)然如果分配的對象是C++中的基本數(shù)據(jù)類型則不會有構(gòu)造函數(shù)調(diào)用。

            如果要深入全局函數(shù)void* ::operator new(size_t)的話,我們會發(fā)現(xiàn),它的具體實現(xiàn)是通過調(diào)用malloc來分配內(nèi)存的,而在win平臺下,malloc最終調(diào)用的是HeapAlloc方法。

            3.6 CoTaskMemAlloc /IMalloc

                 CoTaskMemAlloc用于COM對象,它在進(jìn)程的缺省堆中分配內(nèi)存。

                 IMalloc接口是對 CoTaskMemAlloc/CoTaskMemFree 的再次封裝。

             

             

            posted on 2011-08-12 11:26 肥仔 閱讀(2784) 評論(0)  編輯 收藏 引用 所屬分類: Windows開發(fā)

            国产精品美女久久久免费| 国产精品亚洲综合专区片高清久久久 | 色综合久久久久久久久五月| 99久久婷婷国产综合亚洲| 久久久久国产一级毛片高清板| 亚洲国产另类久久久精品| 精品国产综合区久久久久久| 日韩精品久久久久久久电影蜜臀| 久久久中文字幕| 久久亚洲国产成人精品无码区| 精品人妻久久久久久888| 久久久久国产精品三级网| 国产精品成人99久久久久| 久久人人爽爽爽人久久久| 亚洲а∨天堂久久精品| 国产精品成人久久久久三级午夜电影 | 久久香蕉超碰97国产精品| 无码人妻少妇久久中文字幕 | 好属妞这里只有精品久久| 奇米影视7777久久精品人人爽| 99久久国产主播综合精品| 九九精品99久久久香蕉| 国产A级毛片久久久精品毛片| 久久亚洲精品无码观看不卡| 99精品久久精品| 粉嫩小泬无遮挡久久久久久| 色妞色综合久久夜夜| 久久久久人妻一区二区三区 | 国产精品乱码久久久久久软件| 久久国产香蕉一区精品| 国产成人精品久久| 天天躁日日躁狠狠久久| 久久综合久久自在自线精品自 | 久久综合狠狠综合久久97色| 99久久亚洲综合精品网站| 久久精品国产福利国产秒| 精品综合久久久久久97超人 | av国内精品久久久久影院| 韩国免费A级毛片久久| 亚洲精品美女久久777777| 久久亚洲色一区二区三区|