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