在計算機應用中,經常需要查看二進制文件的內容。目前,在各種VC
++書籍中介紹查看文本文件的文章很多,但鮮有介紹查看二進制文件的文章。本文從功能設計、方案設計、編程實現以及技術要點等方面來簡單介紹如何用VC
++60 查看二進制文件。
一、功能設計
顯示界面將窗口客戶區劃分為三部分,左邊列用于以16
進制方式顯示文件內容的相應位置,中間列用于以16
進制方式顯示文件內容,右邊列用于顯示文件內容對應的ASCII
碼的內容。為簡化程序設計,沒有打印功能。
二、方案設計
采用MFC 的SDI(單文檔界面)。由于在一屏內一般不可能顯示整個文件的內容,所以選擇視類的基類為CScrollView。
二進制文件的讀出與處理在文檔類中完成,文件的顯示與滾動由視類來實現。
三、編程實現
1. 使用MFC AppWizard 向導產生一應用框架
在VC ++的“File"菜單中,單擊“New",彈出一New
對話框。在“Projects"頁中選擇“MFC AppWizard [exe]",在“Project
name"編輯框中填入“HexShow",按“OK"按鈕,退出New
對話框。
在“MFC AppWizard Step 1"對話框中選擇單選鈕“Single document",按“Next>"按鈕,進入“MFC
AppWizard Step 2 of 6"對話框,保持缺省選擇,按“Next>"按鈕,進入“MFC
AppWizard Step 3 of 6"對話框,保持缺省選擇,按“Next>"按鈕,進入“MFC
AppWizard Step 4 of 6"對話框,取消“Printing and print preview"選項,按“Next>"按鈕,進入“MFC
AppWizard Step 5 of 6"對話框,保持缺省設置,繼續按“Next>"按鈕,進入“MFC
AppWizard Step 6 of 6"對話框,在“Base class"組合框中選擇CscrollView,按“Finish"按鈕即可完成應用框架的定制。
2. 在文檔類CHexShowDoc 中增加文件的讀出及處理工作
(1)定義文檔的成員變量,做好初始化及清理工作
打開HexShowDoc.h文件,增加2個公共變量:
CFile *m_pHexFile;
LONG m_lFileLength;
int m_nBytesPerLine;//每行顯示多少個Byte
然后,打開HexShowDoc.cpp文件,
在類的構造函數中增加下列初始化代碼:
m_pHexFile = NULL;
m_lFileLength = 0L;
m_nBytesPerLine=16;// 每行顯示16 個Byte
在類的析構函數中增加下列清理代碼:
if (m_pHexFile != NULL)
{
m_pHexFile ->Close();
delete m_pHexFile;
m_pHexFile = NULL;
}
(2) 在OnOpenDocument()中打開文檔
首先利用ClassWizard重載消息成員函數
OnOpenDocument()。在該成員函數的代碼添加處增加下列代碼:
if (m_pHexFile != NULL)
{
m_pHexFile ->Close();
delete m_pHexFile;
}
m_pHexFile = new CFile(lpszPathName,
CFile::modeRead | CFile::typeBinary);
if (!m_pHexFile)
{
AfxMessageBox("該文件打開錯");
return FALSE;
}
m_lFileLength = m_pHexFile ->GetLength();
(3)增加用于讀文件及進行輸出
<br>格式化處理的成員函數
為CHexShowDoc類增加成員函數如下:
BOOL CHexShowDoc::ReadFileAndProcess
(CString &strLine, LONG lOffset)
{
LONG lPos;
if (lOffset != -1L)
lPos=m_pHexFile ->Seek(lOffset,CFile::begin);
else
lPos = m_pHexFile ->GetPosition();
unsigned char szBuf[16];
int nRet = m_pHexFile ->Read(szBuf, m_nBytesPerLine);
if (nRet <= 0)
return FALSE;
CString sTemp;
CString sChars;
sTemp.Format(_T("%8.8lX : "), lPos);
strLine = sTemp;
for (int i = 0; i < nRet; i ++)
{
if (i == 0)
sTemp.Format(_T("%2.2X"), szBuf[i]);
else if (i %16 == 0)
sTemp.Format(_T("= %2.2X"), szBuf[i]);
else if (i %8 == 0)
sTemp.Format(_T("-%2.2X"), szBuf[i]);
else
sTemp.Format(_T("%2.2X"), szBuf[i]);
if (_istprint(szBuf[i]))
sChars += szBuf[i];
else
sChars += _T('.');
strLine += sTemp;
}
if (nRet < m_nBytesPerLine)
{
CString sPad(_T(''), 2 +3 *(m_nBytesPerLine ?
nRet));
strLine += sPad;
}
strLine += _T(" ");
strLine += sChars;
return TRUE;
}
3. 在視中添加顯示文件內容以及處理滾動操作
(1)變量定義以及初始化:
a)打開HexShowView.h文件,增加成員變量如下:
CFont*m_pFont;//用于為顯示文件內容選擇字體
b)在視的構造函數中為文件內容顯示選擇合適的字體
//選擇一種名為“Fixedsys"的字體,
該字體使得字符的排列整齊
LOGFONT m_logfont;
memset( &m_logfont, 0, sizeof(m_logfont));
_tcscpy(m_logfont.lfFaceName, _T("Fixedsys"));
CClientDC dc(NULL);
m_logfont.lfHeight=::MulDiv(120, dc.GetDevice ?
Caps(LOGPIXELSY), 720);
m_logfont.lfPitchAndFamily = FIXED_PITCH;
m_pFont = new CFont;
m_pFont ->CreateFontIndirect( &m_logfont);
c) 將ChexShowView::OnInitialUpdate()
d) 中的代碼修改為:
CScrollView::OnInitialUpdate();
CHexShowDoc *pDoc = GetDocument();
ASSERT_VALID(pDoc);
CSize sizeTotal(0, pDoc ->m_lFileLength);
SetScrollSizes(MM_TEXT, sizeTotal);
e)在視的析構函數中完成對字體對象的刪除,增加代碼如下:
if (m_pFont != NULL)
delete m_pFont;
(2)在視中的OnDraw()中添加如下代碼:
CHexShowDoc *pDoc = GetDocument();
ASSERT_VALID(pDoc);
CFont * pOldFont;
CString sLine; //用于顯示的文本行
CSize ScrolledSize; //窗口的客戶區的范圍
int iStartLine;//當前屏第一行顯示的行的索引號
int nHeight;//輸出文本行的高度
Crect ScrollRect;
//獲得該屏滾動條的位置
CPoint ScrolledPos=GetScrollPosition();
CRect rectClient;
GetClientRect( &rectClient);
//求出每行的高度(單位:象素數)
TEXTMETRIC tm;//tm用于存儲庫存字體的參數;
pDC ->GetTextMetrics( &tm);
nHeight = tm.tmHeight;
pOldFont = pDC ->SelectObject(m_pFont);
// 根據滾動,求出開始行
ScrolledSize = CSize(rectClient.Width(),
rectClient.Height());
ScrollRect = CRect(rectClient.left,
ScrolledPos.y,rectClient.right, <br>ScrolledSize.cy +
ScrolledPos.y);
iStartLine = ScrolledPos.y/16;
// make sure we are drawing where we should
ScrollRect.top = iStartLine *nHeight;
if (pDoc ->m_pHexFile != NULL)
{
int nLine;
for (nLine = iStartLine; ScrollRect.top <
ScrollRect.bottom; nLine ++)
{
if (!pDoc ->ReadFileAndProcess
(sLine, nLine *16))
break;
nHeight = pDC ->DrawText(sLine,-1, &ScrollRect, <br>DT_TOP |
DT_NOPREFIX | DT_SINGLELINE);
ScrollRect.top += nHeight;
}
}
pDC ->SelectObject(pOldFont);
4. 對該工程進行編譯、連接,形成運行文件HexShow.exe
四、技術關鍵
通過上面介紹,可知該程序并不復雜。其涉及到的技術關鍵有4
條。
1. 利用文檔/
視架構能有效地降低軟件的復雜度,使文檔專注于處理數據,而視由于繼承自CScrollView,則便于文本的顯示和滾動;
2.
選擇一種合適的字體非常重要,否則,可能出現顯示混亂的情況;
3.
選擇一個正確的成員函數往往能起到事半功倍的效果,比如,進行文本輸出時,使用CDC::DrawText(...),就比使用常規的CDC::TextOut(...)
有很大的優點;
4. 不管滾動條處于什么位置,視只顯示所涉及到的文本行?! ?/p>