实用技巧如何用VC++60编写查看二进制文件程序
在计算机应用中,经常需要查看二进制文件的内容。目前,在各种VC++书籍中介绍查看文本文件的文章很多,但鲜有介绍查看二进制文件的文章。本文从功能设计、方案设计、编程实现以及技术要点等方面来简单介绍。
1 功能设计显示界面见图1(略),将窗口客户区划分为三部分,左边列用于以16进制方式显示文件内容的相应位置,中间列用于以16进制方式显示文件内容,右边列用于显示文件内容对应的ASCII码的内容。为简化程序设计,没有打印功能。
2 方案设计采用MFC的SDI(单文档界面)。由于在一屏内一般不可能显示整个文件的内容,所以选择视类的基类为CScrollView。二进制文件的读出与处理在文档类中完成,文件的显示与滚动由视类来实现。
3 编程实现
3.1使用MFCAppWizard向导产生一应用框架在VC++的“File”菜单中,单击“New”,弹出一New对话框。在“Projects”页中选择“MFCAppWizard[exe]”,在“Projectname”编辑框中填入“HexShow”(见图2(略)),按“OK”按钮,退出New对话框。在“MFCAppWizardStep1”对话框中选择单选钮“Singledocument”,按“Next>”按钮,进入“MFCAppWizardStep2of6”对话框,保持缺省选择,按“Next>”按钮,进入“MFCAppWizardStep3of6”对话框,保持缺省选择,按“Next>”按钮,进入“MFCAppWizardStep4of6”对话框,取消“Printingandprintpreview”选项(见图3(略)),按“Next>”按钮,进入“MFCAppWizardStep5of6”对话框,保持缺省设置,继续按“Next>”按钮,进入“MFCAppWizardStep6of6”对话框,在“Baseclass”组合框中选择CscrollView(见图4(略)),按“Finish”按钮即可完成应用框架的定制。
3.2在文档类CHexShowDoc中增加文件的读出及处理工作3.2.1定义文档的成员变量,做好初始化及清理工作打开HexShowDoc.h文件,增加2个公共变量:CFile*m_pHexFile;LONGm_lFileLength;intm_nBytesPerLine;//每行显示多少个Byte然后,打开HexShowDoc.cpp文件,在类的构造函数中增加下列初始化代码:m_pHexFile=NULL;m_lFileLength=0L;m_nBytesPerLine=16;//每行显示16个Byte在类的析构函数中增加下列清理代码:if(m_pHexFile!=NULL){m_pHexFile->Close();deletem_pHexFile;m_pHexFile=NULL;}3.2.2在OnOpenDocument()中打开文档首先利用ClassWizard重载消息成员函数OnOpenDocument()。在该成员函数的代码添加处增加下列代码:if(m_pHexFile!=NULL){m_pHexFile->Close();deletem_pHexFile;}m_pHexFile=newCFile(lpszPathName,CFile::modeRead|CFile::typeBinary);if(!m_pHexFile){AfxMessageBox("该文件打开错");returnFALSE;}m_lFileLength=m_pHexFile->GetLength();3.2.3增加用于读文件及进行输出格式化处理的成员函数为CHexShowDoc类增加成员函数如下:BOOLCHexShowDoc::ReadFileAndProcess(CString&strLine,LONGlOffset){LONGlPos;if(lOffset!=-1L)lPos=m_pHexFile->Seek(lOffset,CFile::begin);elselPos=m_pHexFile->GetPosition();unsignedcharszBuf[16];intnRet=m_pHexFile->Read(szBuf,m_nBytesPerLine);if(nRet<=0)returnFALSE;CStringsTemp;CStringsChars;sTemp.Format(_T("%8.8lX:"),lPos);strLine=sTemp;for(inti=0;i<nRet;i++){if(i==0)sTemp.Format(_T("%2.2X"),szBuf[i]);elseif(i==0)sTemp.Format(_T("=%2.2X"),szBuf[i]);elseif(i%8==0)sTemp.Format(_T("-%2.2X"),szBuf[i]);elsesTemp.Format(_T("%2.2X"),szBuf[i]);if(_istprint(szBuf[i]))sChars+=szBuf[i];elsesChars+=_T('.');strLine+=sTemp;}if(nRet<m_nBytesPerLine){CStringsPad(_T(''),2+3*(m_nBytesPerLine-nRet));strLine+=sPad;}strLine+=_T("");strLine+=sChars;returnTRUE;}
3.3在视中添加显示文件内容以及处理滚动操作3.3.1变量定义以及初始化a)打开HexShowView.h文件,增加成员变量如下:CFont*m_pFont;//用于为显示文件内容选择字体b)在视的构造函数中为文件内容显示选择合适的字体//选择一种名为“Fixedsys”的字体,该字体使得字符的排列整齐LOGFONTm_logfont;memset(&m_logfont,0,sizeof(m_logfont));_tcscpy(m_logfont.lfFaceName,_T("Fixedsys"));CClientDCdc(NULL);m_logfont.lfHeight=::MulDiv(120,dc.GetDeviceCaps(LOGPIXELSY),720);m_logfont.lfPitchAndFamily=FIXED_PITCH;m_pFont=newCFont;m_pFont->CreateFontIndirect(&m_logfont);c)将ChexShowView::OnInitialUpdate()中的代码修改为:CScrollView::OnInitialUpdate();CHexShowDoc*pDoc=GetDocument();ASSERT_VALID(pDoc);CSizesizeTotal(0,pDoc->m_lFileLength);SetScrollSizes(MM_TEXT,sizeTotal);c)在视的析构函数中完成对字体对象的删除,增加代码如下:if(m_pFont!=NULL)deletem_pFont;3.3.2在视中的OnDraw()中添加如下代码CHexShowDoc*pDoc=GetDocument();ASSERT_VALID(pDoc);CFont*pOldFont;CStringsLine;//用于显示的文本行CSizeScrolledSize;//窗口的客户区的范围intiStartLine;//当前屏第一行显示的行的索引号intnHeight;//输出文本行的高度CRectScrollRect;//获得该屏滚动条的位置CPointScrolledPos=GetScrollPosition();CRectrectClient;GetClientRect(&rectClient);//求出每行的高度(单位:象素数)TEXTMETRICtm;//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,ScrolledSize.cy+ScrolledPos.y);iStartLine=ScrolledPos.y/16;//makesurewearedrawingwhereweshouldScrollRect.top=iStartLine*nHeight;if(pDoc->m_pHexFile!=NULL){intnLine;for(nLine=iStartLine;ScrollRect.top<ScrollRect.bottom;nLine++){if(!pDoc->ReadFileAndProcess(sLine,nLine*16))break;nHeight=pDC->DrawText(sLine,-1,&ScrollRect,DT_TOP|DT_NOPREFIX|DT_SINGLELINE);ScrollRect.top+=nHeight;}}pDC->SelectObject(pOldFont);
3.4对该工程进行编译、连接,形成运行文件HexShow.exe经过运行、实际测试,使用效果良好。
4技术关键通过上面介绍,可知该程序并不复杂。其设计到的技术关键有4条。a)利用文档/视架构能有效地降低软件的复杂度,使文档专注于处理数据,而视由于继承自CScrollView,则便于文本的显示和滚动;b)选择一种合适的字体非常重要,否则,可能出现显示混乱的情况;c)选择一个正确的成员函数往往能起到事半功倍的效果,比如,进行文本输出时,使用CDC::DrawText(…),就比使用常规的CDC::TextOut(…)有很大的优点;d)不管滚动条处于什么位置,视只显示所涉及到的文本行。