如何用VC++60编写查看二进制文件程序

 

雷霆工作室 韩燕

 

在计算机应用中,经常需要查看二进制文件的内容。目前,在各种VC++书籍中介绍查看文本文件的文章很多,但鲜有介绍查看二进制文件的文章。本文从功能设计、方案设计、编程实现以及技术要点等方面来简单介绍。

1 功能设计

显示界面见图1(略),
将窗口客户区划分为三部分:
 左边列:用于以16进制方式显示文件内容的相应位置,
 中间列:用于以16进制方式显示文件内容,
 右边列:用于显示文件内容对应的ASCII码的内容。为简化程序设计,没有打印功能。

2 方案设计

采用MFC的SDI(单文档界面)。由于在一屏内一般不可能显示整个文件的内容,所以选择视类的基类为CScrollView。二进制文件的读出与处理在文档类中完成,文件的显示与滚动由视类来实现。

3 编程实现

3.1 使用MFC AppWizard向导产生一应用框架
在VC++的“File”菜单中,单击“New”,弹出一New对话框。
在“Projects”页中选择“MFC AppWizard [exe]”,在“Project name”编辑框中填入“HexShow”(见图2(略)),按“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”选项(见图3(略)),按“Next>”按钮,
进入“MFC AppWizard Step 5 of 6”对话框,保持缺省设置,继续按“Next>”按钮,
进入“MFC AppWizard Step 6 of 6”对话框,在“Base class”组合框中选择CscrollView(见图4(略)), 按“Finish”按钮即可完成应用框架的定制。

3.2 在文档类CHexShowDoc中增加文件的读出及处理工作

3.2.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;
}

3.2.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.2.3 增加用于读文件及进行输出格式化处理的成员函数为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.3 在视中添加显示文件内容以及处理滚动操作

3.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.GetDeviceCaps(LOGPIXELSY), 720);
 m_logfont.lfPitchAndFamily = FIXED_PITCH;
 m_pFont = new CFont;
 m_pFont- >CreateFontIndirect(&m_logfont);
 
 c) 将ChexShowView::OnInitialUpdate()中的代码修改为:
 CScrollView::OnInitialUpdate();
 CHexShowDoc* pDoc = GetDocument();
 ASSERT_VALID(pDoc);
 CSize sizeTotal(0, pDoc- >m_lFileLength);
 SetScrollSizes(MM_TEXT, sizeTotal);

 d) 在视的析构函数中完成对字体对象的删除,增加代码如下:

 if (m_pFont != NULL)
  delete m_pFont;

3.3.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, 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, 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) 不管滚动条处于什么位置,视只显示所涉及到的文本行。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值