來(lái)源:http://www.pconline.com.cn/pcedu/empolder/gj/vc/0509/699389.html
本人在學(xué)習(xí)《WINDOWS核心編程》的時(shí)候?qū)EFFREY大師提到的一個(gè)小程序?qū)懥藘蓚€(gè)版本來(lái)比較性能,該程序的原始需求是這樣的:對(duì)一個(gè)大文件進(jìn)行倒序,也就是將一個(gè)文件頭變成尾,尾變成頭。
使用的方法有很多種,這里使用兩個(gè)方法來(lái)比較,主要是突出使用內(nèi)存映射文件好處;兩種方法為:內(nèi)存映射文件方法,I/O讀寫(xiě)的緩存辦法。
第一種辦法是創(chuàng)建內(nèi)存映射文件對(duì)象,然后將該對(duì)象映射到進(jìn)程的地址空間中,再讀取文件內(nèi)容,然后倒序,再寫(xiě)入文件。
第二中方法是,將文件內(nèi)容讀入一個(gè)大的緩沖區(qū),然后倒序,再寫(xiě)入文件,中間對(duì)原來(lái)的文件刪除,然后重新寫(xiě)入。
程序編寫(xiě)如下
第一種方法,
內(nèi)存映射文件方式:
BOOL FileReverse(PCTSTR pszPathName)


{
HANDLE hFile = CreateFile(pszPathName,GENERIC_WRITE|GENERIC_READ,0,NULL
,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);
if(hFile == INVALID_HANDLE_VALUE)


{
printf("File could not be opened.");
return FALSE;
}

DWORD dwFileSize = GetFileSize(hFile,NULL);

HANDLE hFileMap = CreateFileMapping(hFile,NULL,PAGE_READWRITE,0,
dwFileSize+sizeof(char),NULL);


if(hFileMap == NULL)
{
CloseHandle(hFile);
return FALSE;
}

PVOID pvFile = MapViewOfFile(hFileMap,FILE_MAP_WRITE,0,0,0);


if(pvFile == NULL)
{
CloseHandle(hFileMap);
CloseHandle(hFile);
return FALSE;
}

PSTR pchAnsi = (PSTR)pvFile;
pchAnsi[dwFileSize/sizeof(char)]=0;
_strrev(pchAnsi);

pchAnsi = strchr(pchAnsi,'\n');

while(pchAnsi != NULL)
{
*pchAnsi++ ='\r';
*pchAnsi++ ='\n';
pchAnsi = strchr(pchAnsi,'\n');
}

UnmapViewOfFile(pvFile);
CloseHandle(hFileMap);

SetFilePointer(hFile,dwFileSize,NULL,FILE_BEGIN);
SetEndOfFile(hFile);//實(shí)際上不需要寫(xiě)入了。
CloseHandle(hFile);

return TRUE;
}

第二中方法,
使用緩存的方式:
BOOL FileReverseNoMap(PCTSTR pszPathName)


{
HANDLE hFile = CreateFile(pszPathName,GENERIC_WRITE|GENERIC_READ,0,NULL
,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);
if(hFile == INVALID_HANDLE_VALUE)


{
printf("File could not be opened.");
return FALSE;
}

DWORD dwFileSize = GetFileSize(hFile,NULL);
//CloseHandle(hFile);
char *readBuf = new char[dwFileSize+1];
DWORD nRead = 0,nRet =0;

while(nRead<dwFileSize)
{
if(ReadFile(hFile,readBuf+nRead,dwFileSize-nRead,&nRet,NULL) ==TRUE)


{
nRead+= nRet;
}
else


{
printf("Can read the file!");
CloseHandle(hFile);
}
}

PSTR pchAnsi = (PSTR)readBuf;
pchAnsi[dwFileSize/sizeof(char)]=0;
_strrev(pchAnsi);

pchAnsi = strchr(pchAnsi,'\n');

while(pchAnsi != NULL)
{
*pchAnsi++ ='\r';
*pchAnsi++ ='\n';
pchAnsi = strchr(pchAnsi,'\n');
}
CloseHandle(hFile);
DeleteFile(pszPathName);

HANDLE hWriteFile = CreateFile(pszPathName,GENERIC_WRITE|GENERIC_READ,0,NULL
,CREATE_NEW,FILE_ATTRIBUTE_NORMAL,NULL);
WriteFile(hWriteFile,readBuf,dwFileSize,&nRet,NULL);
CloseHandle(hWriteFile);

delete readBuf;

return TRUE;
}
我運(yùn)行了幾次,比較結(jié)果如下:
文件大小(byte) |
1方法時(shí)間(ms) |
2方法時(shí)間(ms) |
25416 |
0 |
0 |
101664 |
0 |
0 |
406656 |
0 |
10 |
1219968 |
10 |
30 |
3202416 |
21 |
100 |
9607248 |
80 |
551 |
67250736 |
581 |
5568 |
本人測(cè)試機(jī)器的CPU是迅池1.5的筆記本,內(nèi)存為712MB
通過(guò)上面的測(cè)試我們可以看到使用內(nèi)存映射文件的好處,在文件內(nèi)存越大這種優(yōu)勢(shì)就體現(xiàn)的越明顯,其中主要的原因是:
內(nèi)存映射文件直接將文件的地址映射到進(jìn)程的地址空間中,那么操作文件就相當(dāng)于在內(nèi)存中操作一樣,省去了讀和寫(xiě)I/O的時(shí)間;第二種方式是必須這么做(READFILE,WRITEFILE),這個(gè)過(guò)程是很慢的。