用VC++60 查看二进制文件

发表于:2007-07-04来源:作者:点击数: 标签:
雷霆工作室 韩 燕 在计算机应用中,经常需要查看二进制文件的内容。目前,在各种VC ++书籍中介绍查看文本文件的文章很多,但鲜有介绍查看二进制文件的文章。本文从功能设计、方案设计、编程实现以及技术要点等方面来简单介绍如何用VC ++60 查看二进制文件
雷霆工作室 韩 燕

  在计算机应用中,经常需要查看二进制文件的内容。目前,在各种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. 不管滚动条处于什么位置,视只显示所涉及到的文本行。  

原文转自:http://www.ltesting.net