1)

2)

3)

添加CDIB类时,如果没有你要选的那个类,可以先随便选个基类继承,然后自己在代码里把基类修改成要继承的,把一些消息映射的注释掉就可以了,这样的话在建立类向导里也可以找到新建的类。

// TODO: Add your command handler code here
static char szFilter[]="BMP文件(*.bmp)|*bmp||";
//定义过滤文件的类型
CFileDialog dlg(TRUE,"bmp",NULL,OFN_HIDEREADONLY|OFN_OVERWRITEPROMPT,sz
Filter); //定义对话框对象
if(ret==IDOK)
{
filename=dlg.GetFileName(); //获取所选择图像的路径
m_dib.LoadFromFile(filename);//加载图像
if(!m_dib.m_bLoaded) //判断是否加载图像成功
{
AfxMessageBox(图像打不开);
return;
}
}
Invalidate(1);

Visual C++ MFC中没有提供一个专门的类来处理DIB位图,因此,为了方便地使用位图文件,我们有必要派生一个CDib类。类的源代码如下:

  (1) CDib类的声明

// DIB.h:类CDib声明头文件
#ifndef __DIB_H__
#define __DIB_H__
#include <wingdi.h>
class CDib
{
 public:
  CDib();
  ~CDib();

  BOOL Load( const char * );
  BOOL Save( const char * );
  BOOL Draw( CDC *, int nX = 0, int nY = 0, int nWidth = -1, int nHeight = -1, int mode = SRCCOPY);
  BOOL SetPalette( CDC * );

 private:
  CPalette m_Palette;
  unsigned char *m_pDib, *m_pDibBits;
  DWORD m_dwDibSize;
  BITMAPINFOHEADER *m_pBIH;
  RGBQUAD *m_pPalette;
  int m_nPaletteEntries;
};
#endif

  (2) CDib类的实现

// DIB.cpp:类CDib实现文件
#include "stdafx.h"
#include "DIB.h"

CDib::CDib()
{
 m_pDib = NULL;
}

CDib::~CDib()
{
 // 如果位图已经被加载,释放内存
 if (m_pDib != NULL)
  delete []m_pDib;
}

  下面这个函数非常重要,其功能为加载位图,类似于CBitmap类的LoadBitmap函数:

BOOL CDib::Load(const char *pszFilename)
{
 CFile cf;

 // 打开位图文件
 if (!cf.Open(pszFilename, CFile::modeRead))
  return (FALSE);

 // 获得位图文件大小,并减去BITMAPFILEHEADER的长度
 DWORD dwDibSize;
 dwDibSize = cf.GetLength() - sizeof(BITMAPFILEHEADER);

 // 为DIB位图分配内存
 unsigned char *pDib;
 pDib = new unsigned char[dwDibSize];
 if (pDib == NULL)
  return (FALSE);

 BITMAPFILEHEADER BFH;

 // 读取位图文件数据
 try
 {
  // 文件格式是否正确有效
  if ( cf.Read(&BFH, sizeof(BITMAPFILEHEADER)) != sizeof(BITMAPFILEHEADER) ||
     BFH.bfType != ’MB’ || cf.Read(pDib, dwDibSize) != dwDibSize)
  {
   delete []pDib;
   return (FALSE);
  }
 }
 catch (CFileException *e)
 {
  e->Delete();
  delete []pDib;
  return (FALSE);
 }

 // delete先前加载的位图
 if (m_pDib != NULL)
  delete m_pDib;

 // 将临时Dib数据指针和Dib大小变量赋给类成员变量
 m_pDib = pDib;
 m_dwDibSize = dwDibSize;

 // 为相应类成员变量赋BITMAPINFOHEADER和调色板指针
 m_pBIH = (BITMAPINFOHEADER*)m_pDib;
 m_pPalette = (RGBQUAD*) &m_pDib[sizeof(BITMAPINFOHEADER)];

 // 计算调色板中实际颜色数量
 m_nPaletteEntries = 1 << m_pBIH->biBitCount;
 if (m_pBIH->biBitCount > 8)
  m_nPaletteEntries = 0;
 else if (m_pBIH->biClrUsed != 0)
  m_nPaletteEntries = m_pBIH->biClrUsed;

 // 为相应类成员变量赋image data指针
 m_pDibBits = &m_pDib[sizeof(BITMAPINFOHEADER) + m_nPaletteEntries * sizeof (RGBQUAD)];

 // delete先前的调色板
 if (m_Palette.GetSafeHandle() != NULL)
  m_Palette.DeleteObject();

 // 如果位图中存在调色板,创建LOGPALETTE 及CPalette
 if (m_nPaletteEntries != 0)
 {
  LOGPALETTE *pLogPal = (LOGPALETTE*)new char[sizeof(LOGPALETTE) + m_nPaletteEntries *sizeof(PALETTEENTRY)];

  if (pLogPal != NULL)
  {
   pLogPal->palVersion = 0x300;
   pLogPal->palNumEntries = m_nPaletteEntries;

   for (int i = 0; i < m_nPaletteEntries; i++)
   {
    pLogPal->palPalEntry[i].peRed = m_pPalette[i].rgbRed;
    pLogPal->palPalEntry[i].peGreen = m_pPalette[i].rgbGreen;
    pLogPal->palPalEntry[i].peBlue = m_pPalette[i].rgbBlue;
   }

   //创建CPalette并释放LOGPALETTE的内存
   m_Palette.CreatePalette(pLogPal);
   delete []pLogPal;
  }
 }

 return (TRUE);
}

//函数功能:保存位图入BMP文件
BOOL CDib::Save(const char *pszFilename)
{
 if (m_pDib == NULL)
  return (FALSE);

 CFile cf;
 if (!cf.Open(pszFilename, CFile::modeCreate | CFile::modeWrite))
  return (FALSE);

 try
 {
  BITMAPFILEHEADER BFH;
  memset(&BFH, 0, sizeof(BITMAPFILEHEADER));
  BFH.bfType = ’MB’;
  BFH.bfSize = sizeof(BITMAPFILEHEADER) + m_dwDibSize;
  BFH.bfOffBits = sizeof(BITMAPFILEHEADER) +
sizeof(BITMAPINFOHEADER) + m_nPaletteEntries *sizeof(RGBQUAD);

  cf.Write(&BFH, sizeof(BITMAPFILEHEADER));
  cf.Write(m_pDib, m_dwDibSize);
 }
 catch (CFileException *e)
 {
  e->Delete();
  return (FALSE);
 }
 return (TRUE);
}

  下面这个函数也非常重要,其功能为在pDC指向的CDC中绘制位图,起点坐标为(nX,nY),绘制宽度和高度为nWidth、nHeight,最后一个参数是光栅模式:

BOOL CDib::Draw(CDC *pDC, int nX, int nY, int nWidth, int nHeight, int mode)
{
 if (m_pDib == NULL)
  return (FALSE);

 // 获取位图宽度和高度赋值
 if (nWidth == - 1)
  nWidth = m_pBIH->biWidth;
 if (nHeight == - 1)
  nHeight = m_pBIH->biHeight;

 // 绘制位图
 StretchDIBits(pDC->m_hDC, nX, nY, nWidth, nHeight, 0, 0, m_pBIH->biWidth, m_pBIH->biHeight, m_pDibBits, (BITMAPINFO*)m_pBIH, BI_RGB, mode);

 return (TRUE);
}

//函数功能:设置调色板
BOOL CDib::SetPalette(CDC *pDC)
{
 if (m_pDib == NULL)
  return (FALSE);

 // 检查当前是否有一个调色板句柄,对于大于256色的位图,为NULL
 if (m_Palette.GetSafeHandle() == NULL)
  return (TRUE);

 // 选择调色板,接着实施之,最后恢复老的调色板
 CPalette *pOldPalette;
 pOldPalette = pDC->SelectPalette(&m_Palette, FALSE);
 pDC->RealizePalette();
 pDC->SelectPalette(pOldPalette, FALSE);

 return (TRUE);
}

  从整个CDib类的代码中我们可以看出,DIB位图的显示需遵循如下步骤:

  (1)读取位图,本类中使用pDib = new unsigned char[dwDibSize]为位图中的信息分配内存,另一种方法是调用API函数CreateDIBSection,譬如:

m_hBitmap = ::CreateDIBSection(pDC->GetSafeHdc(),
(LPBITMAPINFO) m_lpBMPHdr, DIB_RGB_COLORS,
(LPVOID*) &m_lpDIBits, NULL, 0);

  m_hBitmap定义为:

HBITMAP m_hBitmap;

  (2)根据读取的位图信息,计算出调色板大小,然后创建调色板;

  (3)调用CDib::SetPalette( CDC *pDC )设置调色板,需要用到CDC::SelectPalette及CDC::RealizePalette两个函数;

  (4)调用CDib::Draw(CDC *pDC, int nX, int nY, int nWidth, int nHeight, int mode)函数绘制位图。在此函数中,真正发挥显示位图作用的是对StretchDIBits API函数的调用。StretchDIBits函数具有缩放功能,其最后一个参数也是光栅操作的模式。

  下面给出DIB位图的打开及显示并在其中加入天极网logo的函数源代码。"DIB位图"父菜单下"打开"子菜单的单击事件消息处理函数为(其功能为打开位图并显示之):

void CBitMapExampleDlg::OnOpendibpic()
{
 // 弹出文件对话框,让用户选择位图文件
 CFileDialog fileDialog(TRUE, "*.BMP", NULL, NULL,"位图文件(*.BMP)|*.bmp;*.BMP|");
 if (IDOK == fileDialog.DoModal())
 {
  // 加载位图并显示之
  CDib dib;
  if (dib.Load(fileDialog.GetPathName()))
  {
   CClientDC dc(this);
   dib.SetPalette(&dc);
   dib.Draw(&dc);
  }
 }
}

  "DIB位图"父菜单下"标记"子菜单的单击事件消息处理函数为(其功能为给位图加上天极网logo):

void CBitMapExampleDlg::OnMarkDibpic()
{
 // 弹出文件对话框,让用户选择标记logo
 CFileDialog fileDialog(TRUE, "*.BMP", NULL, NULL, "标记位图文件(*.BMP)|*.bmp;*.BMP|");
 if (IDOK == fileDialog.DoModal())
 {
  // 加载标记logo位图并与目标位图相与
  CDib dib;
  if (dib.Load(fileDialog.GetPathName()))
  {
   CClientDC dc(this);
   dib.SetPalette(&dc);
   dib.Draw(&dc, 0, 0, - 1, - 1, SRCAND);
  }
 }

VC++6.0进行数字图像处理的步骤以及遇到的问题的更多相关文章

  1. 《HALCON数字图像处理》第一、二章笔记

    目录 第一章 绪论 1.1 图像和图像处理 1.1.1 图像 1.1.2 数字图像 1.1.3 图像处理及其发展过程 1.2 数字图像处理的步骤和方法 1.3 数字图像处理系统的硬件组成 1.4 数字 ...

  2. 数字图像处理:基于MATLAB的车牌识别项目 标签: 图像处理matlab算法 2017-06-24 09:17 98人阅读 评论(0)

    学过了数字图像处理,就进行一个综合性强的小项目来巩固一下知识吧.前阵子编写调试了一套基于MATLAB的车牌识别的项目的代码.今天又重新改进了一下代码,识别的效果好一点了,也精简了一些代码.这里没有使用 ...

  3. 数字图像处理实验(总计23个)汇总 标签: 图像处理MATLAB 2017-05-31 10:30 175人阅读 评论(0)

    以下这些实验中的代码全部是我自己编写调试通过的,到此,最后进行一下汇总. 数字图像处理实验(1):PROJECT 02-01, Image Printing Program Based on Half ...

  4. 安装vc++6.0的步骤

    我们学习计算机,就必须要先将编程的c语言学好,打好基础,学习c语言最好的方法就是多上机联系,对于联系我们需要在自己的电脑上安装vc++6.0来进行平日里的联系.1.打开电脑进行联网,打开浏览器搜索vc ...

  5. 【数字图像处理】五.MFC图像点运算之灰度线性变化、灰度非线性变化、阈值化和均衡化处理具体解释

    本文主要讲述基于VC++6.0 MFC图像处理的应用知识,主要结合自己大三所学课程<数字图像处理>及课件进行解说.主要通过MFC单文档视图实现显示BMP图片点运算处理.包含图像灰度线性变换 ...

  6. VC++6.0 编写插件(图文并茂)

    下午偶然注意到VC++6.0新建工程标签页下的DevStudio Add-in Wizard,没有接触过,看名字是给Developer Studio开发插件,心生喜感,于是百度之,发生百度检索几乎找不 ...

  7. 数字图像处理(MATLAB版)学习笔记(2)——第2章 灰度变换与空间滤波

    0.小叙闲言 1.本章整体结构 2.书中例子 例2.1 主要是使用函数imadjust,来熟悉一下灰度处理,体验一把 >> imread('myimage.jpg'); >> ...

  8. 数字图像处理的Matlab实现(1)—绪论

    第1章 绪论 1.1 什么是数字图像处理 一幅图像可以定义为一个二维函数\(f(x,y)\),这里的\(x\)和\(y\)是空间坐标,而在任意坐标\((x,y)\)处的幅度\(f\)被称为这一坐标位置 ...

  9. 数字图像处理笔记与体会(一)——matlab编程基础

    最近开始学习数字图像处理,使用matlab实现,下面我就来记录笔记和体会,一方面是给大家提供参考,另一方面是防止我忘记了. 复习一下: 1.数字图像是用一个数字矩阵来表示的,数字阵列中的每个数字,表示 ...

随机推荐

  1. Feign 负载均衡

    一.是什么 Feign 是一个声明式 WebService 客户端.使用 Feign 能让编写 Web Service 客户端更加简单,他的使用方法是定义一个接口,然后在上面添加注解.同时也支持 JA ...

  2. 转载 - 跳跃的舞者,舞蹈链(Dancing Links)算法——求解精确覆盖问题

    出处:http://www.cnblogs.com/grenet/p/3145800.html 精确覆盖问题的定义:给定一个由0-1组成的矩阵,是否能找到一个行的集合,使得集合中每一列都恰好包含一个1 ...

  3. 一次获取多个oracle序列值

    一次获取多个oracle序列值 学习了:http://blog.csdn.net/wangchsh2008/article/details/53495961 select seq_one.nextva ...

  4. codechef Little Elephant and Bombs题解

    The Little Elephant from the Zoo of Lviv currently is on the military mission. There are N enemy bui ...

  5. [Cypress] Test React’s Controlled Input with Cypress Selector Playground

    React based applications often use controlled inputs, meaning the input event leads to the applicati ...

  6. 图论-BFS解无权有向图最短路径距离

    概述 本篇博客主要内容: 对广度优先搜索算法(Breadth-First-Search)进行介绍: 介绍用邻接表的存储结构实现一个图(附C++实现源代码): 介绍用BFS算法求解无权有向图(附C++实 ...

  7. 【JavaScript】在同一个网页中实现多个JavaScript特效

    在网页中,假设出现两次<script type="text/javascript"></script>标签,全部的JavaScipt脚本都不会再生效,仅仅能 ...

  8. laravel接口设计

    在各种公共方法都设计好,软件安装成功的条件下 routes/web.php中路由信息如下 <?php /* |------------------------------------------ ...

  9. 0x52 背包

    还行 前面的题没啥意思 完全背包一个很难写的题poj1015 Jury Compromise poj1742 多重背包啊,开始写了二进制拆分TLE了,由于是判是否可以组成,那么可以用一个贪心的思想,顺 ...

  10. Swift - 将字符串拆分成数组(把一个字符串分割成字符串数组)

    在Swift中,如果需要把一个字符串根据特定的分隔符拆分(split)成字符串数组,通常有如下两种方法: 1,使用componentsSeparatedByString()方法 1 2 3 4 5 l ...