• <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>

            C++ Programmer's Cookbook

            {C++ 基礎(chǔ)} {C++ 高級(jí)} {C#界面,C++核心算法} {設(shè)計(jì)模式} {C#基礎(chǔ)}

            windows核心編程--虛擬內(nèi)存

            Wi n d o w s提供了3種進(jìn)行內(nèi)存管理的方法,它們是:
            虛擬內(nèi)存,最適合用來管理大型對(duì)象或結(jié)構(gòu)數(shù)組。

            內(nèi)存映射文件,最適合用來管理大型數(shù)據(jù)流(通常來自文件)以及在單個(gè)計(jì)算機(jī)上運(yùn)行的多個(gè)進(jìn)程之間共享數(shù)據(jù)。

            內(nèi)存堆棧,最適合用來管理大量的小對(duì)象。


            虛擬內(nèi)存的狀態(tài)
            Wi n d o w s函數(shù)G l o b a l M e m o r y S t a t u s可用于檢索關(guān)于當(dāng)前內(nèi)存狀態(tài)的動(dòng)態(tài)信息:
            VOID GlobalMemoryStatus(LPMEMORYSTATUS pmst);
            如果希望應(yīng)用程序在內(nèi)存大于4 G B的計(jì)算機(jī)上運(yùn)行,或者合計(jì)交換文件的大小大于4 G B,那么可以使用新的G l o b a l M e m o r y S t a t u s E x函數(shù):
            BOOL GlobalMemoryStatusEx(LPMEMORYSTATUSEX pmst);

            確定地址空間的狀態(tài)
            Wi n d o w s提供了一個(gè)函數(shù),可以用來查詢地址空間中內(nèi)存地址的某些信息(如大小,存儲(chǔ)器類型和保護(hù)屬性等)。
            這個(gè)函數(shù)稱為Vi r t u a l Q u e r y:
            DWORD VirtualQuery(
               LPCVOID pvAddress,
               PMEMORY_BASIC_INFORMATION pmbi,
               DWORD dwLength);
            Wi n d o w s還提供了另一個(gè)函數(shù),它使一個(gè)進(jìn)程能夠查詢另一個(gè)進(jìn)程的內(nèi)存信息:
            DWORD VirtualQueryEx(
               HANDLE hProcess,
               LPCVOID pvAddress,
               PMEMORY_BASIC_INFORMATION pmbi,
               DWORD dwLength);
            這兩個(gè)函數(shù)基本相同,差別在于使用Vi r t u a l Q u e r y E x時(shí),可以傳遞你想要查詢的地址空間信息的進(jìn)程的句柄。調(diào)試程序和其他實(shí)用程序使用這個(gè)函數(shù)最多,幾乎所有的應(yīng)用程序都只需要調(diào)用Vi r t u a l Q u e r y函數(shù)。
            為了獲得完整的內(nèi)存信息,我創(chuàng)建了一個(gè)函數(shù),即V M Q u e r y:
            BOOL VMQuery(
               HANDLE hProcess,
               PVOID pvAddress,
               PVMQUERY pVMQ);
            系統(tǒng)信息

            許多操作系統(tǒng)的值是根據(jù)主機(jī)而定的,比如頁面的大小,分配粒度的大小等。這些值決不應(yīng)該用硬編碼的形式放入你的源代碼。相反,你始終都應(yīng)該在進(jìn)程初始化的時(shí)候檢索這些值,并在你的源代碼中使用檢索到的值。G e t S y s t e m I n f o函數(shù)將用于檢索與主機(jī)相關(guān)的值:

             

            VOID GetSystemInfo(LPSYSTEM_INFO psinf);


            必須傳遞S Y S T E M _ I N F O結(jié)構(gòu)的地址給這個(gè)函數(shù)。這個(gè)函數(shù)將初始化所有的結(jié)構(gòu)成員然后返回。下面是S Y S T E M _ I N F O數(shù)據(jù)結(jié)構(gòu)的樣子。

             

            typedef  struct  _SYSTEM_INFO
            {
               union
               
            {
                  DWORD dwOemId;  
            // Obsolete, do not use
                   struct  
                  
            {
                     WORD wProcessorArchitecture;
                     WORD wReserved;
                  }
            ;
               }
            ;
               DWORD     dwPageSize;
               LPVOID    lpMinimumApplicationAddress;
               LPVOID    lpMaximumApplicationAddress;
               DWORD_PTR dwActiveProcessorMask;
               DWORD     dwNumberOfProcessors;
               DWORD     dwProcessorType;
               DWORD     dwAllocationGranularity;
               WORD      wProcessorLevel;
               WORD      wProcessorRevision;
            }
             SYSTEM_INFO,  * LPSYSTEM_INFO;


            當(dāng)系統(tǒng)引導(dǎo)時(shí),它要確定這些成員的值是什么。對(duì)于任何既定的系統(tǒng)來說,這些值總是相同的,因此決不需要為任何既定的進(jìn)程多次調(diào)用該函數(shù)。

            表14-1 與內(nèi)存有關(guān)的成員函數(shù)

            成員名 描述
            d w P a g e S i z e 用于顯示C P U的頁面大小。在x86 CPU上,這個(gè)值是4 0 9 6字節(jié)。在Alpha CPU 上,這個(gè)值是8 1 9 2字節(jié)。在I A - 6 4上,這個(gè)值是8 1 9 2字節(jié)
            l p M i n i m u m A p p l i c a t i o n A d d r e s s 用于給出每個(gè)進(jìn)程的可用地址空間的最小內(nèi)存地址。在Windows 98上,這個(gè)值是4 194 304,或0 x 0 0 4 0 0 0 0 0,因?yàn)槊總€(gè)進(jìn)程的地址空間中下面的4 M B是不能使用的。在Windows 2000上,這個(gè)值是65 536或0 x 0 0 0 1 0 0 0 0,因?yàn)槊總€(gè)進(jìn)程的地址空間中開頭的6 4 K B總是空閑的
            l p M a x i m u m A p p l i c a t i o n A d d r e s s 用于給出每個(gè)進(jìn)程的可用地址空間的最大內(nèi)存地址。在Windows 98 上,這個(gè)地址是2 147 483 647或0 x 7 F F F F F F F,因?yàn)楣蚕韮?nèi)存映射文件區(qū)域和共享操作系統(tǒng)代碼包含在上面的2 GB分區(qū)中。在Windows 2000上,這個(gè)地址是內(nèi)核方式內(nèi)存開始的地址,它不足6 4 K B
            d w A l l o c a t i o n G r a n u l a r i t y 顯示保留的地址空間區(qū)域的分配粒度。截止到撰寫本書時(shí),在所有Wi n d o w s平臺(tái)上,這個(gè)值都是65536

            該結(jié)構(gòu)的其他成員與內(nèi)存管理毫無關(guān)系,為了完整起見,下面也對(duì)它們進(jìn)行了介紹(見表1 4 - 2)。

             

            表14-2 與內(nèi)存無關(guān)的成員函數(shù)

            成員名 描述
            d w O e m I d 已作廢,不引用
            W R e d e r v e d 保留供將來使用,不引用
            d w N u m b e r O f P r o c e s s o r s 用于指明計(jì)算機(jī)中的C P U數(shù)目
            d w A c t i v e P r o c e s s o r M a s k 一個(gè)位屏蔽,用于指明哪個(gè)C P U是活動(dòng)的(允許運(yùn)行線程)
            d w P r o c e s s o r Ty p e 只用于Windows 98,不用于Windows 2000,用于指明處理器的類型,如Intel 386、4 8 6或P e n t i u m
            w P r o c e s s o r A r c h i t e c t u r e 只用于Windows 2000,不用于Windows 98,用于指明處理的結(jié)構(gòu),如I n t e l、A l p h a、Intel 64位或Alpha 64位
            w P r o c e s s o r L e v e l 只用于Windows 2000,不用于Windows 98,用于進(jìn)一步細(xì)分處理器的結(jié)構(gòu),如用于設(shè)定Intel Pentium Pro或Pentium II
            w P r o c e s s o r R e v i s i o n 只用于Windows 2000 ,不用于Windows 98,用于進(jìn)一步細(xì)分處理器的級(jí)別

            在地址空間中保留一個(gè)區(qū)域

            通過調(diào)用Vi r t u a l A l l o c函數(shù),可以在進(jìn)程的地址空間中保留一個(gè)區(qū)域:

             

            PVOID VirtualAlloc(
            PVOID pvAddress,
            SIZE_T dwSize,
            DWORD fdwAllocationType,
            DWORD fdwProtect);

            第一個(gè)參數(shù)p v A d d r e s s包含一個(gè)內(nèi)存地址,用于設(shè)定想讓系統(tǒng)將地址空間保留在什么地方。在大多數(shù)情況下,你為該參數(shù)傳遞M U L L。它告訴Vi r t u a l A l l o c,保存著一個(gè)空閑地址區(qū)域的記錄的系統(tǒng)應(yīng)該將區(qū)域保留在它認(rèn)為合適的任何地方。系統(tǒng)可以從進(jìn)程的地址空間的任何位置來保留一個(gè)區(qū)域,因?yàn)椴荒鼙WC系統(tǒng)可以從地址空間的底部向上或者從上面向底部來分配各個(gè)區(qū)域。可以使用M E M _ TO P _ D O W N標(biāo)志來說明該分配方式。如果Vi r t u a l A l l o c函數(shù)能夠滿足你的要求,那么它就返回一個(gè)值,指明保留區(qū)域的基地址。如果傳遞一個(gè)特定的地址作為Vi r t u a l A l l o c的p v A d d r e s s 參數(shù),那么該返回值與傳遞給Vi r t u a l A l l o c的值相同,并被圓整為(如果需要的話) 6 4 K B邊界值。

            Vi r t u a l A l l o c函數(shù)的第二個(gè)參數(shù)是d w S i z e,用于設(shè)定想保留的區(qū)域的大小(以字節(jié)為計(jì)量單位)。由于系統(tǒng)保留的區(qū)域始終必須是C P U頁面大小的倍數(shù),因此,如果試圖保留一個(gè)跨越6 2 K B的區(qū)域,結(jié)果就會(huì)在使用4 KB、8 KB或16 KB頁面的計(jì)算機(jī)上產(chǎn)生一個(gè)跨越6 4 K B的區(qū)域。

            Vi r t u a l A l l o c函數(shù)的第三個(gè)參數(shù)是f d w A l l o c a t i o n Ty p e,它能夠告訴系統(tǒng)你想保留一個(gè)區(qū)域還是提交物理存儲(chǔ)器(這樣的區(qū)分是必要的,因?yàn)閂i r t u a l A l l o c函數(shù)也可以用來提交物理存儲(chǔ)器)。若要保留一個(gè)地址空間區(qū)域,必須傳遞M E M _ R E S E RV E標(biāo)識(shí)符作為F d w A l l o c a t i o n Ty p e參數(shù)的值。


            最后一個(gè)參數(shù)是f d w P r o t e c t,用于指明應(yīng)該賦予該地址空間區(qū)域的保護(hù)屬性。與該區(qū)域相關(guān)聯(lián)的保護(hù)屬性對(duì)映射到該區(qū)域的已提交內(nèi)存沒有影響。無論賦予區(qū)域的保護(hù)屬性是什么,如果沒有提交任何物理存儲(chǔ)器,那么訪問該范圍中的內(nèi)存地址的任何企圖都將導(dǎo)致該線程引發(fā)一個(gè)訪問違規(guī)。

            在保留區(qū)域中的提交存儲(chǔ)器

            當(dāng)保留一個(gè)區(qū)域后,必須將物理存儲(chǔ)器提交給該區(qū)域,然后才能訪問該區(qū)域中包含的內(nèi)存地址。系統(tǒng)從它的頁文件中將已提交的物理存儲(chǔ)器分配給一個(gè)區(qū)域。物理存儲(chǔ)器總是按頁面邊界和頁面大小的塊來提交的。

            若要提交物理存儲(chǔ)器,必須再次調(diào)用Vi r t u a l A l l o c函數(shù)。不過這次為f d w A l l o c a t i o n Ty p e參數(shù)傳遞的是M E M _ C O M M I T標(biāo)志,而不是M E M _ R E S E RV E標(biāo)志。傳遞的頁面保護(hù)屬性通常與調(diào)用Vi r t u a l A l l o c來保留區(qū)域時(shí)使用的保護(hù)屬性相同(大多數(shù)情況下是PA G E _ R E A D W R I T E),不過也可以設(shè)定一個(gè)不同的保護(hù)屬性。

            在已保留的區(qū)域中,你必須告訴Vi r t u a l A l l o c函數(shù),你想將物理存儲(chǔ)器提交到何處,以及要提交多少物理存儲(chǔ)器。為了做到這一點(diǎn),可以在p v A d d r e s s參數(shù)中設(shè)定你需要的內(nèi)存地址,并在d w S i z e參數(shù)中設(shè)定物理存儲(chǔ)器的數(shù)量(以字節(jié)為計(jì)量單位)。注意,不必立即將物理存儲(chǔ)器提交給整個(gè)區(qū)域。
            同時(shí)進(jìn)行區(qū)域的保留和內(nèi)存的提交

            有時(shí)你可能想要在保留區(qū)域的同時(shí),將物理存儲(chǔ)器提交給它。只需要一次調(diào)用Vi r t u a l A l l o c函數(shù)就能進(jìn)行這樣的操作,如下所示:

             

            PVOID pvMem  =  VirtualAlloc(NULL,  99   *   1024 ,   MEM_RESERVE  |  MEM_COMMIT, PAGE_READWRITE);


            這個(gè)函數(shù)調(diào)用請(qǐng)求保留一個(gè)99 KB的區(qū)域,并且將99 KB的物理存儲(chǔ)器提交給它。當(dāng)系統(tǒng)處理這個(gè)函數(shù)調(diào)用時(shí),它首先要搜索你的進(jìn)程的地址空間,找出未保留的地址空間中一個(gè)地址連續(xù)的區(qū)域,它必須足夠大,能夠存放100 KB(在4 KB頁面的計(jì)算機(jī)上)或104 KB(在8 KB頁面的計(jì)算機(jī)上)。

            系統(tǒng)之所以要搜索地址空間,原因是已將p v A d d r e s s參數(shù)設(shè)定為N U L L。如果為p v A d d r e s s設(shè)定了內(nèi)存地址,系統(tǒng)就要查看在該內(nèi)存地址上是否存在足夠大的未保留地址空間。如果系統(tǒng)找不到足夠大的未保留地址空間,Vi r t u a l A l l o c將返回N U L L,

            如果能夠保留一個(gè)合適的區(qū)域,系統(tǒng)就將物理存儲(chǔ)器提交給整個(gè)區(qū)域。無論是該區(qū)域還是提交的內(nèi)存,都將被賦予PA G E _ R E A D W R I T E保護(hù)屬性。

            最后需要說明的是,Vi r t u a l A l l o c將返回保留區(qū)域和提交區(qū)域的虛擬地址,然后該虛擬地址被保存在p v M e m變量中。如果系統(tǒng)無法找到足夠大的地址空間,或者不能提交該物理存儲(chǔ)器,Vi r t u a l A l l o c將返回N U L L。
            回收虛擬內(nèi)存和釋放地址空間區(qū)域

            若要回收映射到一個(gè)區(qū)域的物理存儲(chǔ)器,或者釋放這個(gè)地址空間區(qū)域,可調(diào)用Vi r t u a l F r e e函數(shù):

             

            BOOL VirtualFree(
            LPVOID pvAddress,
            SIZE_T dwSize,
            DWORD fdwFreeType);

            改變保護(hù)屬性

            雖然實(shí)踐中很少這樣做,但是可以改變已經(jīng)提交的物理存儲(chǔ)器的一個(gè)或多個(gè)頁面的保護(hù)屬性。

            若要改變內(nèi)存頁面的保護(hù)屬性,可以調(diào)用Vi r t u a l P r o t e c t函數(shù):

             

                BOOL VirtualProtect(
               PVOID pvAddress,
               SIZE_T dwSize,
               DWORD flNewProtect,
               PDWORD pflOldProtect);


            當(dāng)然,保護(hù)屬性是與內(nèi)存的整個(gè)頁面相關(guān)聯(lián)的,而不是賦予內(nèi)存的各個(gè)字節(jié)的。因此,如果要使用下面的代碼來調(diào)用4 KB 頁面的計(jì)算機(jī)上的Vi r t u a l P r o t e c t函數(shù),其結(jié)果是把PA G E _ N O A C C E S S保護(hù)屬性賦予內(nèi)存的兩個(gè)頁面:

             

            VirtualProtect(pvRgnBase + (3 * 1024), 2 * 1024,
            PAGE_NOACCESS, &flOldProtect);

            清除物理存儲(chǔ)器的內(nèi)容


            為了說明內(nèi)存的內(nèi)容已經(jīng)被清除,我們必須對(duì)系統(tǒng)的R A M提出大量的使用需求。若要進(jìn)行這項(xiàng)操作,可以分3步來進(jìn)行:

            1) 調(diào)用G l o b a l M e m o r y S t a t u s函數(shù),獲取計(jì)算機(jī)中R A M的總?cè)萘俊?/font>

            2) 調(diào)用Vi r t u a l A l l o c函數(shù),提交該數(shù)量的內(nèi)存。這項(xiàng)操作的運(yùn)行速度非常快,因?yàn)樵谶M(jìn)程試圖訪問頁面之前,系統(tǒng)實(shí)際上并不為該內(nèi)存分配R A M。

            3) 調(diào)用Z e r o M e m o r y函數(shù),使新提交的頁面可以被訪問。這將給系統(tǒng)的R A M帶來沉重的負(fù)擔(dān),導(dǎo)致當(dāng)前正在R A M中的某些頁面被寫入頁文件。

            如果用戶指明該數(shù)據(jù)將在以后被訪問,那么該數(shù)據(jù)將不被清除,并且在以后訪問該數(shù)據(jù)時(shí)將數(shù)據(jù)轉(zhuǎn)入R A M。但是,如果用戶指明以后將不再訪問該數(shù)據(jù),那么數(shù)據(jù)將被清除,并且系統(tǒng)不把數(shù)據(jù)寫入頁文件,這樣就可以提高應(yīng)用程序的運(yùn)行性能。



             

            posted on 2006-09-20 18:12 夢(mèng)在天涯 閱讀(3634) 評(píng)論(0)  編輯 收藏 引用 所屬分類: Windows API

            公告

            EMail:itech001#126.com

            導(dǎo)航

            統(tǒng)計(jì)

            • 隨筆 - 461
            • 文章 - 4
            • 評(píng)論 - 746
            • 引用 - 0

            常用鏈接

            隨筆分類

            隨筆檔案

            收藏夾

            Blogs

            c#(csharp)

            C++(cpp)

            Enlish

            Forums(bbs)

            My self

            Often go

            Useful Webs

            Xml/Uml/html

            搜索

            •  

            積分與排名

            • 積分 - 1807719
            • 排名 - 5

            最新評(píng)論

            閱讀排行榜

            偷窥少妇久久久久久久久| 无码乱码观看精品久久| 伊人久久亚洲综合影院| 无码人妻少妇久久中文字幕| 91超碰碰碰碰久久久久久综合| 久久久久久久久波多野高潮| 国产激情久久久久久熟女老人| 欧美日韩精品久久久久| 久久久精品一区二区三区| 久久久久综合国产欧美一区二区| 久久亚洲sm情趣捆绑调教 | 久久久久av无码免费网| 久久夜色精品国产噜噜亚洲AV| 91久久九九无码成人网站| 一本久道久久综合狠狠躁AV| 日本强好片久久久久久AAA| 国产香蕉97碰碰久久人人| 日韩精品久久无码人妻中文字幕| 26uuu久久五月天| 狠狠色综合网站久久久久久久高清 | 无码人妻久久一区二区三区免费 | 久久精品免费一区二区| 色综合久久中文色婷婷| 新狼窝色AV性久久久久久| 久久久久国产精品嫩草影院 | 婷婷久久精品国产| 久久久国产精品福利免费| 亚洲精品乱码久久久久66| 蜜臀久久99精品久久久久久| 国产成人久久激情91| 伊人久久大香线蕉av不变影院| 久久亚洲国产成人影院网站 | 国产情侣久久久久aⅴ免费| 狠狠色综合网站久久久久久久高清| 久久精品国产亚洲av瑜伽| 一本一道久久精品综合| 久久97精品久久久久久久不卡| 亚洲AV日韩AV天堂久久| 亚洲综合熟女久久久30p| 久久久久久精品久久久久| 2020国产成人久久精品|