📄 bmp灰度图像平滑处理.txt
字号:
一、用VC++6.0实现BMP图象文件的显示
1. 用VC建立一个项目(MFC App Winzard(exe)),选择单文档视图,其余默认。
2. 为了将BMP中的数据读入内存,在项目中建立专门处理BMP文件头和数据的文件:DIBAPI.H 和DIBAPI.CPP,数据文件已提供(自己参考使用)。
3. 在Doc 类中添加CPalette * m_palDIB, HDIB m_hDIB,CSize m_sizeDoc 分别保存调色板,图象句柄和文档大小。添加方法:GetHDIB,GetDocPalette,GetDocSize
HDIB GetHDIB()const
{return m_hDIB;}
CPalette *GetDocPalette() const
{return m_palDIB;}
CSize GetDocSize() const
{return m_sizeDoc;}
添加方法:InitDIBData()
{
// 初始化DIB对象
// 判断调色板是否为空
if (m_palDIB != NULL)
{
// 删除调色板对象
delete m_palDIB;
// 重置调色板为空
m_palDIB = NULL;
}
// 如果DIB对象为空,直接返回
if (m_hDIB == NULL)
{
// 返回
return;
}
LPSTR lpDIB = (LPSTR) ::GlobalLock((HGLOBAL) m_hDIB);
// 判断图像是否过大
if (::DIBWidth(lpDIB) > INT_MAX ||::DIBHeight(lpDIB) > INT_MAX)
{
::GlobalUnlock((HGLOBAL) m_hDIB);
// 释放DIB对象
::GlobalFree((HGLOBAL) m_hDIB);
// 设置DIB为空
m_hDIB = NULL;
CString strMsg;
strMsg = "BMP图像太大!";
// 提示用户
MessageBox(NULL, strMsg, "系统提示", MB_ICONINFORMATION | MB_OK);
// 返回
return;
}
// 设置文档大小
m_sizeDoc = CSize((int) ::DIBWidth(lpDIB), (int) ::DIBHeight(lpDIB));
::GlobalUnlock((HGLOBAL) m_hDIB);
// 创建新调色板
m_palDIB = new CPalette;
// 判断是否创建成功
if (m_palDIB == NULL)
{
// 失败,可能是内存不足
::GlobalFree((HGLOBAL) m_hDIB);
// 设置DIB对象为空
m_hDIB = NULL;
// 返回
return;
}
// 调用CreateDIBPalette来创建调色板
if (::CreateDIBPalette(m_hDIB, m_palDIB) == NULL)
{
// 返回空,可能该DIB对象没有调色板
// 删除
delete m_palDIB;
// 设置为空
m_palDIB = NULL;
// 返回
return;
}
}
4.响应类Doc 的OnOpenDocument 事件,以实现打开文件的操作。
BOOL CCh1_1Doc::OnOpenDocument(LPCTSTR lpszPathName) //111
{
if (!CDocument::OnOpenDocument(lpszPathName))
return FALSE;
CFile file;
CFileException fe;
// 打开文件
if (!file.Open(lpszPathName, CFile::modeRead | CFile::shareDenyWrite, &fe))
{
// 失败
ReportSaveLoadException(lpszPathName, &fe,
FALSE, AFX_IDP_FAILED_TO_OPEN_DOC);
// 返回FALSE
return FALSE;
}
DeleteContents();
// 更改光标形状
BeginWaitCursor();
// 尝试调用ReadDIBFile()读取图像
TRY
{
m_hDIB = ::ReadDIBFile(file);
}
CATCH (CFileException, eLoad)
{
// 读取失败
file.Abort();
// 恢复光标形状
EndWaitCursor();
// 报告失败
ReportSaveLoadException(lpszPathName, eLoad,
FALSE, AFX_IDP_FAILED_TO_OPEN_DOC);
// 设置DIB为空
m_hDIB = NULL;
// 返回FALSE
return FALSE;
}
END_CATCH
// 初始化DIB
InitDIBData();
// 恢复光标形状
EndWaitCursor();
// 判断读取文件是否成功
if (m_hDIB == NULL)
{
// 失败,可能非BMP格式
CString strMsg;
strMsg = "读取图像时出错!可能是不支持该类型的图像文件!";
// 提示出错
MessageBox(NULL, strMsg, "系统提示", MB_ICONINFORMATION | MB_OK);
// 返回FALSE
return FALSE;
}
// 设置文件名称
SetPathName(lpszPathName);
// 初始化胀标记为FALSE
SetModifiedFlag(FALSE);
// 返回TRUE
return TRUE;
}
5. 响应类Doc 的OnSaveDocument 事件,完成保存图象的操作。
BOOL CCh1_1Doc::OnSaveDocument(LPCTSTR lpszPathName) //111
{
CFile file;
CFileException fe;
// 打开文件
if (!file.Open(lpszPathName, CFile::modeCreate |
CFile::modeReadWrite | CFile::shareExclusive, &fe))
{
// 失败
ReportSaveLoadException(lpszPathName, &fe,
TRUE, AFX_IDP_INVALID_FILENAME);
// 返回FALSE
return FALSE;
}
// 尝试调用SaveDIB保存图像
BOOL bSuccess = FALSE;
TRY
{
// 更改光标形状
BeginWaitCursor();
// 尝试保存图像
bSuccess = ::SaveDIB(m_hDIB, file);
// 关闭文件
file.Close();
}
CATCH (CException, eSave)
{
// 失败
file.Abort();
// 恢复光标形状
EndWaitCursor();
ReportSaveLoadException(lpszPathName, eSave,
TRUE, AFX_IDP_FAILED_TO_SAVE_DOC);
// 返回FALSE
return FALSE;
}
END_CATCH
// 恢复光标形状
EndWaitCursor();
// 重置胀标记为FALSE
SetModifiedFlag(FALSE);
if (!bSuccess)
{
// 保存失败,可能是其它格式的DIB,可以读取但是不能保存
// 或者是SaveDIB函数有误
CString strMsg;
strMsg = "无法保存BMP图像!";
// 提示出错
MessageBox(NULL, strMsg, "系统提示", MB_ICONINFORMATION | MB_OK);
}
return bSuccess;
}
6. 响应类View的OnDraw事件,完成显示的操作。
void CCh1_1View::OnDraw(CDC* pDC) //111
{
// 显示等待光标
BeginWaitCursor();
// 获取文档
CCh1_1Doc* pDoc = GetDocument();
ASSERT_VALID(pDoc);
// 获取DIB
HDIB hDIB = pDoc->GetHDIB();
// 判断DIB是否为空
if (hDIB != NULL)
{
LPSTR lpDIB = (LPSTR) ::GlobalLock((HGLOBAL) hDIB);
// 获取DIB宽度
int cxDIB = (int) ::DIBWidth(lpDIB);
// 获取DIB高度
int cyDIB = (int) ::DIBHeight(lpDIB);
::GlobalUnlock((HGLOBAL) hDIB);
CRect rcDIB;
rcDIB.top = rcDIB.left = 0;
rcDIB.right = cxDIB;
rcDIB.bottom = cyDIB;
CRect rcDest;
// 判断是否是打印
if (pDC->IsPrinting())
{
// 是打印,计算输出图像的位置和大小,以便符合页面
// 获取打印页面的水平宽度(象素)
int cxPage = pDC->GetDeviceCaps(HORZRES);
// 获取打印页面的垂直高度(象素)
int cyPage = pDC->GetDeviceCaps(VERTRES);
// 获取打印机每英寸象素数
int cxInch = pDC->GetDeviceCaps(LOGPIXELSX);
int cyInch = pDC->GetDeviceCaps(LOGPIXELSY);
// 计算打印图像大小(缩放,根据页面宽度调整图像大小)
rcDest.top = rcDest.left = 0;
rcDest.bottom = (int)(((double)cyDIB * cxPage * cyInch)
/ ((double)cxDIB * cxInch));
rcDest.right = cxPage;
// 计算打印图像位置(垂直居中)
int temp = cyPage - (rcDest.bottom - rcDest.top);
rcDest.bottom += temp/2;
rcDest.top += temp/2;
}
else
// 非打印
{
// 不必缩放图像
rcDest = rcDIB;
}
// 输出DIB
::PaintDIB(pDC->m_hDC, &rcDest, pDoc->GetHDIB(),
&rcDIB, pDoc->GetDocPalette());
}
// 恢复正常光标
EndWaitCursor();
}
7. 编译并运行程序。
二、显示256灰度级的灰度直方图
1. 在上个实验程序的基础上,添加一个对话框,该对话框对应的类为:DlgIntensity。要在对话框中绘制出图像的灰度直方图,首先必须得到图像像素的指针(保存在类成员变量m_lpDIBBits中)和图像的高度、宽度信息(保存在类成员变量m_lHeight和m_lWidth中)。有了这些信息,就可以计算出各个灰度的像素数(保存在类成员变量m_lCount[256]数组中)。在显示灰度直方图时,可以指定其灰度的上限和下限。在类中设置两个变量m_iLowGraw,m_iUpGraw来保存。
2. 对话框头文件
#if !defined(AFX_DLGINTENSITY_H__289F4D93_14C7_42BC_837E_3A673E972A92__INCLUDED_)
#define AFX_DLGINTENSITY_H__289F4D93_14C7_42BC_837E_3A673E972A92__INCLUDED_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
// DlgIntensity.h : header file
//
/////////////////////////////////////////////////////////////////////////////
// DlgIntensity dialog
class DlgIntensity : public CDialog
{
// Construction
public:
// 当前鼠标拖动状态,0表示未拖动,1表示正在拖动下限,2表示正在拖动上限。
int m_iIsDraging;
// 相应鼠标事件的矩形区域
CRect m_MouseRect;
// DIB的高度
LONG m_lHeight;
// DIB的宽度
LONG m_lWidth;
// 指向当前DIB象素的指针
char * m_lpDIBBits;
// 各个灰度值的计数
LONG m_lCount[256];
DlgIntensity(CWnd* pParent = NULL); // standard constructor
// Dialog Data
//{{AFX_DATA(DlgIntensity)
enum { IDD = IDD_DIALOG1 };
int m_iLowGray; //下限
int m_iUpGray; //上限
//}}AFX_DATA
// Overrides
// ClassWizard generated virtual function overrides
//{{AFX_VIRTUAL(DlgIntensity)
protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV support
//}}AFX_VIRTUAL
// Implementation
protected:
// Generated message map functions
//{{AFX_MSG(DlgIntensity)
afx_msg void OnPaint();
afx_msg void OnKillfocusEDITLowGray();
afx_msg void OnKillfocusEDITUpGray();
virtual void OnOK();
virtual BOOL OnInitDialog();
afx_msg void OnMouseMove(UINT nFlags, CPoint point);
afx_msg void OnLButtonDown(UINT nFlags, CPoint point);
afx_msg void OnLButtonUp(UINT nFlags, CPoint point);
//}}AFX_MSG
DECLARE_MESSAGE_MAP()
};
//{{AFX_INSERT_LOCATION}}
// Microsoft Visual C++ will insert additional declarations immediately before the previous line.
#endif // !defined(AFX_DLGINTENSITY_H__289F4D93_14C7_42BC_837E_3A673E972A92__INCLUDED_)
3. 对话框程序
// DlgIntensity.cpp : implementation file
//
#include "stdafx.h"
#include "ch1_1.h"
#include "DlgIntensity.h"
#include "DIBAPI.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
/////////////////////////////////////////////////////////////////////////////
// DlgIntensity dialog
DlgIntensity::DlgIntensity(CWnd* pParent /*=NULL*/)
: CDialog(DlgIntensity::IDD, pParent)
{
//{{AFX_DATA_INIT(DlgIntensity)
m_iLowGray = 0;
m_iUpGray = 0;
//}}AFX_DATA_INIT
}
void DlgIntensity::DoDataExchange(CDataExchange* pDX)
{
CDialog::DoDataExchange(pDX);
//{{AFX_DATA_MAP(DlgIntensity)
DDX_Text(pDX, IDC_EDIT_LowGray, m_iLowGray);
DDV_MinMaxInt(pDX, m_iLowGray, 0, 255);
DDX_Text(pDX, IDC_EDIT_UpGray, m_iUpGray);
DDV_MinMaxInt(pDX, m_iUpGray, 0, 255);
//}}AFX_DATA_MAP
}
BEGIN_MESSAGE_MAP(DlgIntensity, CDialog)
//{{AFX_MSG_MAP(DlgIntensity)
ON_WM_PAINT()
ON_EN_KILLFOCUS(IDC_EDIT_LowGray, OnKillfocusEDITLowGray)
ON_EN_KILLFOCUS(IDC_EDIT_UpGray, OnKillfocusEDITUpGray)
ON_WM_MOUSEMOVE()
ON_WM_LBUTTONDOWN()
ON_WM_LBUTTONUP()
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////////////////////
// DlgIntensity message handlers
void DlgIntensity::OnPaint()
{
// 字符串
CString str;
// 循环变量
LONG i;
// 最大计数
LONG lMaxCount = 0;
// 设备上下文
CPaintDC dc(this);
// 获取绘制坐标的文本框
CWnd* pWnd = GetDlgItem(IDC_COORD);
// 指针
CDC* pDC = pWnd->GetDC();
pWnd->Invalidate();
pWnd->UpdateWindow();
pDC->Rectangle(0,0,330,300);
// 创建画笔对象
CPen* pPenRed = new CPen;
// 红色画笔
pPenRed->CreatePen(PS_SOLID,1,RGB(255,0,0));
// 创建画笔对象
CPen* pPenBlue = new CPen;
// 蓝色画笔
pPenBlue->CreatePen(PS_SOLID,1,RGB(0,0, 255));
// 创建画笔对象
CPen* pPenGreen = new CPen;
// 绿色画笔
pPenGreen->CreatePen(PS_DOT,1,RGB(0,255,0));
// 选中当前红色画笔,并保存以前的画笔
CGdiObject* pOldPen = pDC->SelectObject(pPenRed);
// 绘制坐标轴
pDC->MoveTo(10,10);
// 垂直轴
pDC->LineTo(10,280);
// 水平轴
pDC->LineTo(320,280);
// 写X轴刻度值
str.Format("0");
pDC->TextOut(10, 283, str);
str.Format("50");
pDC->TextOut(60, 283, str);
str.Format("100");
pDC->TextOut(110, 283, str);
str.Format("150");
pDC->TextOut(160, 283, str);
str.Format("200");
pDC->TextOut(210, 283, str);
str.Format("255");
pDC->TextOut(265, 283, str);
// 绘制X轴刻度
for (i = 0; i < 256; i += 5)
{
if ((i & 1) == 0)
{
// 10的倍数
pDC->MoveTo(i + 10, 280);
pDC->LineTo(i + 10, 284);
}
else
{
// 10的倍数
pDC->MoveTo(i + 10, 280);
pDC->LineTo(i + 10, 282);
}
}
// 绘制X轴箭头
pDC->MoveTo(315,275);
pDC->LineTo(320,280);
pDC->LineTo(315,285);
// 绘制X轴箭头
pDC->MoveTo(10,10);
pDC->LineTo(5,15);
pDC->MoveTo(10,10);
pDC->LineTo(15,15);
// 计算最大计数值
for (i = m_iLowGray; i <= m_iUpGray; i ++)
{
// 判断是否大于当前最大值
if (m_lCount[i] > lMaxCount)
{
// 更新最大值
lMaxCount = m_lCount[i];
}
}
// 输出最大计数值
pDC->MoveTo(10, 25);
pDC->LineTo(14, 25);
str.Format("%d", lMaxCount);
pDC->TextOut(11, 26, str);
// 更改成绿色画笔
pDC->SelectObject(pPenGreen);
// 绘制窗口上下限
pDC->MoveTo(m_iLowGray + 10, 25);
pDC->LineTo(m_iLowGray + 10, 280);
pDC->MoveTo(m_iUpGray + 10, 25);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -