在 Windows CE 下申請大容量物理內存
申請大容量的物理內存看起來(lái)不是難事。這里的大容量是指幾十MB甚至更多的物理內存。對于C++程序員來(lái)說(shuō)可能平時(shí)習慣了使用"new"操作符來(lái)實(shí)現。我也是這樣。使用"new"非常簡(jiǎn)單,申請之后只需判斷返回的指針是否是空即可。在其它的Windows操作系統上的確不需要在申請大容量物理內存上過(guò)多考慮。但是在Windows CE上就不同了。如果只用"new"就能搞定,那就太省事了。
不知道Windows CE下軟件開(kāi)發(fā)者是否遇到過(guò)這種情況,如果使用"new"申請超過(guò)30MB的物理內存,那么返回的一定是空(NULL),甚至程序會(huì )死鎖無(wú)法響應。這其實(shí)不奇怪。在《Windows CE下進(jìn)程、線(xiàn)程和內存管理》的系列文章中我早有所言,Windows CE下每個(gè)進(jìn)程占有32MB的地址空間,雖然Slot 1槽存放所有的非XIP DLL,但是我們不可能占用Slot 1槽。32MB地址空間減去必要的代碼段、靜態(tài)數據段、默認堆和默認棧之后,所剩的地址空間少于32MB。即使程序什么都不做也無(wú)法滿(mǎn)足超過(guò)30MB的地址空間的申請需求。所以返回為空非常正常。好在Windows CE下運行的大多數軟件不需要那么多的物理內存。
感覺(jué)微軟的技術(shù)不是支持到很遠的將來(lái),而是得過(guò)且過(guò),只要滿(mǎn)足目前的和不遠的將來(lái)的需求就行。拿Platform Builder來(lái)說(shuō),IMGRAM64環(huán)境變量用于支持64MB物理內存??墒菦](méi)有IMGRAM128或者IMGRAM256甚至IMGRAM512??赡苁钱敃r(shí)絕大多數基于Windows CE的產(chǎn)品都沒(méi)有超過(guò)64MB物理內存?,F在要支持超過(guò)64MB物理內存就必須做一些修改操作。再如現在說(shuō)的用"new"分配物理內存,也只是限制在32MB以?xún)?。如果想new多少就new多少,那多爽!
"new"不行是因為地址空間不夠,那我們可以采用虛擬內存分配,然后提交物理內存這種辦法。理論上是這樣,但是實(shí)際上還是不行。舉例如下: LPVOID g_Address1, g_Address2;
g_Address1 = VirtualAlloc(0, 32 * 1024 * 1024, MEM_RESERVE, PAGE_NOACCESS);
g_Address2 = VirtualAlloc(g_Address1, 32 * 1024 * 1024, MEM_COMMIT, PAGE_READWRITE);
上面這段代碼中第一個(gè)語(yǔ)句是申請32MB的虛擬地址空間,函數返回一個(gè)地址說(shuō)明申請是成功的。注意這個(gè)地址一定處于0x4200 0000以上(具體參見(jiàn)我的專(zhuān)欄中“Windows CE下進(jìn)程、線(xiàn)程和內存管理(三)”)。第二個(gè)語(yǔ)句是提交物理內存,容量為32MB。這個(gè)函數返回NULL,說(shuō)明申請物理內存不成功。如果申請10MB、20MB的還可以。
希望再一次破滅。最后的辦法就是內存映射文件了。在Windows CE的幫助文檔中只提到了內存映射文件可以用來(lái)申請虛擬地址空間??梢栽囈辉?。結果證明用內存映射文件來(lái)申請大容量物理內存是可行的。內存映射文件用于多個(gè)進(jìn)程共享數據時(shí),創(chuàng )建內存映射的函數的第一個(gè)參數必須設置為INVALID_HANDLE_VALUE,表示在物理內存中創(chuàng )建。利用這個(gè)特點(diǎn)我們可以申請超過(guò)32MB的物理內存。具體能夠申請的大小由剩余的物理內存決定。例子如下: #define MAXLEN (64*1024*1024)
HANDLE hFile;
hFile = CreateFileMapping(INVALID_HANDLE_VALUE, NULL, PAGE_READWRITE, 0, MAXLEN, NULL);
if(hFile == NULL)
{
//創(chuàng )建文件映射對象失敗
return;
}
LPVOID lpAddress;
lpAddress = MapViewOfFile(hFile, FILE_MAP_WRITE|FILE_MAP_READ, 0, 0, MAXLEN);
if(lpAddress == NULL)
{
//創(chuàng )建文件視圖失敗
return;
}
上述的函數如果都成功了,你就可以使用物理內存了。物理內存的首地址是lpAddress。使用完了別忘了調用函數UnmapViewOfFile(lpAddress); 和CloseHandle(hFile);
聯(lián)系客服