MFC详解显示BMP格式图片

本文主要是讲述《数字图像处理》系列栏目中的第一篇文章.主要详细介绍了BMP图片格式,同时使用C++和MFC显示BMP格式,主要结合自己的《数字图像处理》课程和以前的项目叙述讲解.

一.BMP图片格式定义

BMP文件格式是Windows操作系统推荐和支持的标准图像文件格式,是一种将内存或显示器的图像数据不经过压缩而直接按位存盘的文件格式,故称位图(bitmap),其扩展名为BMP.BMP图像通常有4个部分组成:位图文件头、位图信息头、颜色表、位图数据.如下图所示:

第一部分为位图文件头BITMAPFILEHEADER.位图文件头结构长度固定为14个字节,包含文件的类型、大小、位图文件保留字、位图数据距文件头的偏移量.其中WORD为无符号16位整数(2byte),DWORD为无符号32位整数(4byte).具体结构体定义如下:

 
  1. //位图文件头

  2. typedef struct tagBITMAPFILEHEADER {

  3. WORD bfType; //位图文件的类型,必须为BM 0x424d 表示.bmp

  4. DWORD bfSize; //位图文件的大小,以字节为单位 包括该14字节

  5. WORD bfReserved1; //位图文件保留字,必须为0

  6. WORD bfReserved2; //位图文件保留字,必须为0

  7. DWORD bfOffBits; //位图数据距文件头的偏移量,以字节为单位 即前三部分和

  8. } BITMAPFILEHEADER;

第二部分为位图信息头BITMAPINFOHEADER,该结构也固定为40个字节,用于说明位图的尺寸、宽高、像素、分辨率、颜色表等信息.具体结构定义如下:

 
  1. //位图信息头

  2. typedef struct tagBITMAPINFOHEADER {

  3. DWORD biSize; //本结构所占用字节数 40字节

  4. LONG biWidth; //位图的宽度,以像素为单位

  5. LONG biHeight; //位图的高度,以像素为单位

  6. WORD biPlanes; //目标设备的级别,必须为1

  7. WORD biBitCount; //每个像素所需的位数,必须是1(双色)、

  8. //4(16色)、8(256色)或24(真彩色)之一

  9. DWORD biCompression; //位图压缩类型,必须是 0(BI_RGB不压缩)、

  10. //1(BI_RLE8压缩类型)或2(BI_RLE压缩类型)之一

  11. DWORD biSizeImage; //位图的大小,以字节为单位

  12. LONG biXPelsPerMeter; //位图水平分辨率,每米像素数

  13. LONG biYPelsPerMeter; //位图垂直分辨率,每米像素数

  14. DWORD biClrUsed; //位图实际使用的颜色表中的颜色数

  15. DWORD biClrImportant; //位图显示过程中重要的颜色数

  16. } BITMAPINFOHEADER;

第三部分为颜色表或调色板(Palette).有些位图需要调色板,有些位图如真彩色图(biBitCount=24)不需要调色板,它们的BITMAPINFOHEADER后面直接是位图数据.调色板实际是一个数组,共有biClrUsed个元素(如果该值为零,则有2的biBitCount次幂个元素).数组中每个元素的类型是一个RGBQUAD结构,占4字节.定义如下:

 
  1. //位图颜色表

  2. typedef struct tagRGBQUAD

  3. {

  4. BYTE rgbBlue; //蓝色的亮度(值范围为0~255)

  5. BYTE rgbGreen; //绿色的亮度(值范围为0~255)

  6. BYTE rgbRed; //红色的亮度(值范围为0~255)

  7. BYTE rgbReserved; //保留,必须为0

  8. } RGBQUAD;

第四部分就是实际的图像数据.对于真彩色图(24位位图 biBitCount=24),图像数据就是实际的RGB值;对于用到调色板的位图,图像数据就是该像素颜色在调色板中的索引值.下面对2色、16色、256色和真彩色位图分别介绍:
(1).2色位图:当biBitCount=1时,用1位就可以表示该像素的颜色(0表示黑,1表示白),所以8个像素占1个字节;
(2).16色位图:当biBitCount=4时,用4为可以表示一个像素的颜色,所以2个像素占1个字节;
(3).256色位图:当biBitCount=8时,用1个字节表示1个像素,1个像素占1个字节;
(4).真彩色图:当biBitCount=24时,此时用3个字节表示1个像素,其中RGB各占1字节,由于没有颜色表,位图信息头后面是位图数据.

同时,注意以下几点:
1.由于Windows规定一个扫描所占的字节数必须是4的倍数(即以long为单位),不足的以0填充.同时注意下面公式,计算只含位图数据的大小:
biSizeImage=(((bi.biWidth*bi.biBitCount)+31)/(32*4))*bi.Height
在后面讲述获取文件的信息时会通过UE软件结合16进制数据进行详细讲解上面各个数据的具体含义.
2.BMP图片格式的数据是从下到上、从左到右读.即文件中最先读到的图像是最下面一行的左边第一个元素,即从左下角开始存储(0,0)点,从左下角到右上角存储数据.尤其是在图像几何变换平移、旋转时,我就犯过这样的错误,本想让图像从左下角向右上移动,结果刚好相反,后面也会通过实例加深大家的印象.
3.如果想使用C语言\C++显示图片,建议自定义个ImageStruct.h的头文件.包含BMP位图的位图文件头结构、位图信息头结构、位图颜色表3个结构,在实例变量操作.而使用MFC,因为在wingdi.h文件中系统已经定义了BMP图像的结构BITMAPFILEHEADER、BITMAPINFOHEADER,直接在View.h中用他俩实例定义即可.

二.显示BMP图片的基本步骤

在MFC工程XXXView.h类中添加成员函数void ShowBitmap(CDC* pDC,CString BmpName);通过自定义函数实现显示BMP格式图像,其中*pDC是CDC句柄,BmpName是图像文件名.具体步骤如下:
1.创建位图并调用函数LoadImage装载图标、光标或位图.
  HBITMAP m_hBitmap; 
  m_hBitmap=(HBITMAP)LoadImage(HINSTANCE hinst,LPCTSTR lpszName,UINT uType,int cxDesired,int cyDesired,UINT fuLoad)
2.定义并创建一个内存设备环境DC,调用函数CreateCompatibleDC创建兼容的DC.
  CDC dcBmp; dcBmp.CreateCompatibleDC(pDC) ;
3.定义BITMAP变量,调用函数GetBitmap将图片载入位图中,该定义是为后去图像的长宽等信息.
  BITMAP m_bmp; m_bitmap.GetBitmap(&m_bmp);
4.调用函数SelectObject将位图选入兼容内存设备环境DC中.
  dcBmp.SelectObject(&m_bitmap);
5.将兼容的DC中的位图填到当前DC中,调用函数BitBlt或strechBlt显示图像.
(1).BitBlt()该函数对指定的源设备环境区域中的像素进行位块(bit_block)转换,以传送到目标设备环境.
  pDC->BitBlt(0,0,m_bmp.bmWidth,m_bmp.bmHeight,&dcBmp,0,0,SRCCOPY);
(2).stretchBlt()该函数从源矩形中复制位图到目标矩形,必要是按目标设备设置的模式进行图像拉伸或压缩.
  pDC->StretchBlt(0,0,m_nDrawWidth,m_nDrawHeight,&dcBmp,0,0,m_bmp.bmWidth,m_bmp.bmHeight,SRCCOPY);
6.恢复临时DC的位图,删除CreateCompatibleDC得到的图片DC,删除内存中的位图及释放系统资源.
  dcBmp.SelectObject(pbmpOld); DeleteObject(&m_bitmap);  dcBmp.DeleteDC();
具体函数代码如下:

 
  1. //****************显示BMP格式图片****************//

  2. void CShowBMPView::ShowBitmap(CDC *pDC, CString BmpName)

  3. {

  4. //定义bitmap指针 调用函数LoadImage装载位图

  5. HBITMAP m_hBitmap;

  6. m_hBitmap = (HBITMAP) LoadImage(NULL,BmpName,IMAGE_BITMAP,0,0,LR_LOADFROMFILE|LR_DEFAULTSIZE|LR_CREATEDIBSECTION);

  7.  
  8. /*************************************************************************/

  9. /* 1.要装载OEM图像,则设此参数值为0 OBM_ OEM位图 OIC_OEM图标 OCR_OEM光标

  10. /* 2.BmpName要装载图片的文件名

  11. /* 3.装载图像类型:

  12. /* IMAGE_BITMAP-装载位图 IMAGE_CURSOR-装载光标 IMAGE_ICON-装载图标

  13. /* 4.指定图标或光标的像素宽度和长度 以像素为单位

  14. /* 5.加载选项:

  15. /* IR_LOADFROMFILE-指明由lpszName指定文件中加载图像

  16. /* IR_DEFAULTSIZE-指明使用图像默认大小

  17. /* LR_CREATEDIBSECTION-当uType参数为IMAGE_BITMAP时,创建一个DIB项

  18. /**************************************************************************/

  19.  
  20. if( m_bitmap.m_hObject )

  21. {

  22. m_bitmap.Detach(); //切断CWnd和窗口联系

  23. }

  24. m_bitmap.Attach(m_hBitmap); //将句柄HBITMAP m_hBitmap与CBitmap m_bitmap关联

  25.  
  26. //边界

  27. CRect rect;

  28. GetClientRect(&rect);

  29.  
  30. //图片显示(x,y)起始坐标

  31. int m_showX=0;

  32. int m_showY=0;

  33. int m_nWindowWidth = rect.right - rect.left; //计算客户区宽度

  34. int m_nWindowHeight = rect.bottom - rect.top; //计算客户区高度

  35.  
  36. //定义并创建一个内存设备环境DC

  37. CDC dcBmp;

  38. if( !dcBmp.CreateCompatibleDC(pDC) ) //创建兼容性的DC

  39. return;

  40.  
  41. BITMAP m_bmp; //临时bmp图片变量

  42. m_bitmap.GetBitmap(&m_bmp); //将图片载入位图中

  43.  
  44. CBitmap *pbmpOld = NULL;

  45. dcBmp.SelectObject(&m_bitmap); //将位图选入临时内存设备环境

  46.  
  47. //图片显示调用函数stretchBlt

  48. pDC->StretchBlt(0,0,m_bmp.bmWidth,m_bmp.bmHeight,&dcBmp,0,0,m_bmp.bmWidth,m_bmp.bmHeight,SRCCOPY);

  49.  
  50. /*******************************************************************************/

  51. /* BOOL StretchBlt(int x,int y,int nWidth,int nHeight,CDC* pSrcDC,

  52. /* int xSrc,int ySrc,int nSrcWidth,int nSrcHeight,DWORD dwRop );

  53. /* 1.参数x、y位图目标矩形左上角x、y的坐标值

  54. /* 2.nWidth、nHeigth位图目标矩形的逻辑宽度和高度

  55. /* 3.pSrcDC表示源设备CDC指针

  56. /* 4.xSrc、ySrc表示位图源矩形的左上角的x、y逻辑坐标值

  57. /* 5.dwRop表示显示位图的光栅操作方式 SRCCOPY用于直接将位图复制到目标环境中

  58. /*******************************************************************************/

  59.  
  60. dcBmp.SelectObject(pbmpOld); //恢复临时DC的位图

  61. DeleteObject(&m_bitmap); //删除内存中的位图

  62. dcBmp.DeleteDC(); //删除CreateCompatibleDC得到的图片DC

  63. }

补充:MFC中DC指device context,设备环境或设备描述表,它其实是GDI内部保存数据的一种数据结构.此结构中属性内容与特定输出设备相关,属性定义了GDI函数的工作细节.总之,使用GDI绘图函数,就需要一个DC句柄,MFC中把和DC相关的封装成类.其中CDC是一个抽象基类,可以访问整个显示器和打印机.其中下面链接中的文章详细描述了CBitmap、HBitmap、Bitmap三者的区别与联系.http://blog.csdn.net/ivan_ljf/article/details/8569130

三.MFC显示BMP图片

下面将详细讲解使用VS2012 MFC创建工程的具体步骤:
第一步:新建项目"MFC应用程序",项目名为ShowBMP,在应用程序类型中选择"单个文档",点击"确定".在右栏的"资源视图"中,点击"Menu->IDR_MAINFRAM"可以查看并修改菜单视图.
第二步:向CShowBMPView类添加成员变量和成员函数.在右栏的"类视图"右键CShowBMPView添加函数或直接在ShowBMPView.h中直接添加public成员变量和成员函数.添加代码如下:

 
  1. public:

  2. //成员变量

  3. CString BmpName; //保存图像文件文件名

  4. CString EntName; //保存图像文件扩展名

  5. CBitmap m_bitmap; //创建位图对象

  6.  
  7. //成员函数

  8. void ShowBitmap(CDC* pDC,CString BmpName); //用来显示指定位图bmp的函数

第三步:设置打开BMP图片函数."项目"->"类向导"->选择"类名"CShowBMPView->在命令对象ID中双击"ID_FILE_OPEN"->自动生成默认成员函数OnFileOpen,消息为COMMAND.双击成员函数(Member Functions)进入函数编辑.(VC++ 6.0中Ctrl+W可以实现建立类向导)

向添加成员函数CShowBMPView::OnFileOpen()添加如下代码,主要是生成打开图片的对话框,并获取图片路径及后缀.自定义四种格式为bmp gif jpg tiff,但目前只能打开bmp格式图片.

 
  1. //**************文件打开****************//

  2. void CShowBMPView::OnFileOpen()

  3. {

  4. //四种格式的文件:bmp gif jpg tiff

  5. CString filter;

  6. filter="所有文件(*.bmp,*.jpg,*.gif,*tiff)|*.bmp;*.jpg;*.gif;*.tiff| BMP(*.bmp)|*.bmp| JPG(*.jpg)|*.jpg| GIF(*.gif)|*.gif| TIFF(*.tiff)|*.tiff||";

  7. CFileDialog dlg(TRUE,NULL,NULL,OFN_HIDEREADONLY,filter,NULL);

  8.  
  9. //按下确定按钮 dlg.DoModal() 函数显示对话框

  10. if( dlg.DoModal() == IDOK )

  11. {

  12. BmpName = dlg.GetPathName(); //获取文件路径名 如D:\pic\abc.bmp

  13. EntName = dlg.GetFileExt(); //获取文件扩展名

  14. EntName.MakeLower(); //将文件扩展名转换为一个小写字符

  15. Invalidate(); //调用该函数就会调用OnDraw重绘画图

  16. }

  17. }

第四步:在ShowBMPView.cpp中编写void CShowBMPView::ShowBitmap(CDC *pDC, CString BmpName)函数,即“二.显示BMP图片基本步骤”.同时通过OnDraw()函数调用ShowBitmap()函数显示图片.代码如下:

 
  1. //在OnDraw函数中调用ShowBitmap()实现图片的显示功能

  2. void CShowBMPView::OnDraw(CDC* pDC)

  3. {

  4. CShowBMPDoc* pDoc = GetDocument();

  5. ASSERT_VALID(pDoc);

  6. if (!pDoc)

  7. return;

  8.  
  9. // TODO: 在此处为本机数据添加绘制代码

  10.  
  11. if( EntName.Compare(_T("bmp")) == 0 ) //bmp格式

  12. {

  13. ShowBitmap(pDC,BmpName); //显示图片

  14. }

  15. }

四.运行结果及总结

运行程序后,显示如下所示:其中可以看到自定义的打开对话框和显示图片.

  

最后,该文章主要是数字图像处理的基础知识,详细介绍了BMP图片格式和使用MFC如何读取BMP图片的相关知识.

 

//

转载:https://blog.csdn.net/eastmount/article/details/18238863

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/492325.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

0pencv——图像腐蚀

1、代码如下&#xff1a; #include "stdafx.h" #include <opencv2/opencv.hpp>using namespace cv;int main() {Mat srcImage imread("小狗1.jpg");imshow("显示图像", srcImage);Mat element getStructuringElement(MORPH_RECT, Size(…

腾讯研究院发布:《人工智能+制造产业发展研究》报告

来源&#xff1a;腾讯研究院摘要&#xff1a;工业革命以后的“自动化”概念追求的是机器自动生产&#xff0c;本质是“机器替人”&#xff0c;强调在完全不需要人的情况下进行不间断的大规模机器生产&#xff1b;而“智能化”追求的是机器的柔性生产&#xff0c;本质是“人机协…

Opencv——图像模糊

1、代码如下&#xff1a; #include "stdafx.h" #include <opencv2/opencv.hpp>using namespace cv;int main() {Mat srcImage imread("小狗1.jpg");imshow("原图像", srcImage);Mat dstImage;blur(srcImage, dstImage, Size(5, 5));imsh…

Android开发中依赖注入的应用

什么是依赖注入&#xff1f; 依赖是指一个对象持有其他对象的引用。依赖注入则是将这些依赖对象传递给被依赖对象&#xff0c;而不是被依赖对象自己创建这些对象。 public class MyClass{private AnotherClass mAnotherObject;public MyClass(){mAnotherObject new AnotherCla…

工业富联2018年报来了!上市后首张成绩单大起底

未来智能实验室是人工智能学家与科学院相关机构联合成立的人工智能&#xff0c;互联网和脑科学交叉研究机构。未来智能实验室的主要工作包括&#xff1a;建立AI智能系统智商评测体系&#xff0c;开展世界人工智能智商评测&#xff1b;开展互联网&#xff08;城市&#xff09;云…

Opencv——图像膨胀

1、代码如下&#xff1a; #include "stdafx.h" #include <opencv2/opencv.hpp>using namespace cv;int main() {Mat srcImage imread("小狗1.jpg");imshow("原图像", srcImage);Mat dstImage;Mat element getStructuringElement(MORPH_…

深度学习背后的基础-神经网络揭秘

来源&#xff1a;混沌巡洋舰摘要&#xff1a;最近&#xff0c; 深度学习三杰获得了计算机界最重要的图灵奖&#xff0c; 它们的贡献都集中在对深度学习的根据神经网络的理论突破。 今天我们看到的所有和人工智能有关的伟大成就&#xff0c; 从阿法狗到自动驾驶&#xff0c; 从海…

Opencv——灰度变换、直方图均衡化

1、代码如下&#xff1a; #include "stdafx.h" #include <opencv2/opencv.hpp>using namespace cv;int main() {Mat srcImage imread("lena.bmp");Mat grayImage;cvtColor(srcImage, grayImage, COLOR_BGR2GRAY);Mat dstImage;equalizeHist(grayIm…

并发模型之——共享内存模型(线程与锁)理论篇

这里我们使用Java的线程与锁来解析共享内存模型&#xff1b;做过java开发并且了解线程安全问题的知道&#xff0c;要使某段代码是线程安全的那必须要满足两个条件&#xff1a;内存可见性、原子性&#xff1b; 内存可见性 在JVM规定多个线程进行通讯是通过共享变量进行的&a…

谷歌员工怒了 900人联名抗议 刚成立的AI道德委员会处境尴尬

来源&#xff1a;网易智能谷歌员工又怒了。3月26日&#xff0c;谷歌宣布成立人工智能项目外部顾问委员会&#xff0c;该委员会将与谷歌就面部识别和公平性等人工智能的主要问题进行磋商。争议的焦点在于&#xff0c;谷歌将保守派传统基金会(Heritage Foundation)主席凯科尔斯詹…

Opencv——基于索引表的图像细化

图像细化针对的是二值图像 或者用阀值处理的二值图像。基于索引表的细化算法大致是遍历被二值化图像的边缘&#xff0c;根据边缘点的八连通域情况查找索引表以确定该边缘点是否能够被删除。根据一些细化规则我们可以建立索引表&#xff0c;因此我们的主要工作就是不断地遍历边…

DeepMind推出首个商业产品,30秒内准确诊断眼疾!

来源&#xff1a;Financial Times、智东西编译摘要&#xff1a;这个设备能像最好的医学专家一样&#xff0c;准确地诊断各种眼部疾病。4月1日&#xff0c;谷歌母公司Alphabet旗下位于伦敦的AI部门DeepMind已打造出了可诊断复杂眼部疾病的商业医疗设备原型&#xff0c;这将是Dee…

Opencv——Sobel边缘检测

1、代码如下&#xff1a; #include "stdafx.h" #include <opencv2/opencv.hpp>using namespace cv;int main() {Mat srcImage imread("lena.jpg");Mat dstImage_x, dstImage_y;Sobel(srcImage, dstImage_x, CV_8U, 1, 0);Sobel(srcImage, dstImag…

Head first servlet and jsp学习笔记

学习中遇到的问题&#xff1a;java基础不行&#xff0c;都忘光了。 主要是&#xff1a;继承&#xff0c;接口&#xff0c;多线程&#xff0c;IO。尤其是多线程&#xff0c;在分布式系统中应该使用的比较多 第一章&#xff1a;前言和体系结构 HTTP协议&#xff1a; TCP/IP的上层…

一文解析|首个上榜科创板的机器人企业,江苏北人“闯关记”

来源&#xff1a;机器人大讲堂摘要&#xff1a;随着上交所公布了科创板首批受理上市申请的企业名单&#xff0c;这九家企业的每一家都被拿到放大镜下细细观察&#xff0c;评头论足。而其中&#xff0c;江苏北人作为登上科创版的首家机器人企业似乎受到的关注最多。江苏北人是一…

Matlab——绘制基础曲线

1、代码如下&#xff1a; close all;clear all;clc; %关闭所有图形窗口&#xff0c;清除工作空间所有变量&#xff0c;清空命令行 x0:0.02:10; y1sin(x); y22*sin(x); plot(x,y1,b*:,x,y2,r-); %设置颜色、标记和线型 axis([0 pi 0 2]); %设置坐标轴 title(正弦曲…

2018年中美独角兽研究报告

来源&#xff1a;前瞻产业研究院截至2018年12月31日&#xff0c;中国共有205家独角兽企业&#xff0c;占全球独角兽总量的48%&#xff0c;美国共有149家独角兽企业&#xff0c;占全球独角兽总量的35%。中美独角兽行业估值结构差异明显&#xff0c; 但总体看来&#xff0c;对于中…

Matlab——高斯白噪声处理

1、代码如下&#xff1a; close all;clear all;clc; %关闭所有图形窗口&#xff0c;清除工作空间所有变量&#xff0c;清空命令行 Iimread(eight.tif); Aimnoise(I,gaussian,0,0.05); %加入高斯白噪声 JA; M3; %设置叠加次数 Jim2double(J); Iim2…

中国首份自动驾驶路测报告:自主车企全面落后

来源&#xff1a;车云摘要&#xff1a;首份中国自动驾驶路测报告&#xff0c;透露着科技企业的野心和中国车企的无奈。近日&#xff0c;我国首份自动驾驶车辆道路测试年度报告出炉。这份由北京市发布的自动驾驶道路测试报告中&#xff0c;首次披露了自动驾驶相关企业&#xff0…

19个决定性时刻,2030年前,这些黑科技必将发生

来源&#xff1a;科技联盟技术中心摘要&#xff1a;从无人驾驶汽车到机器人工人&#xff0c;在我们熟悉所有这一切之前&#xff0c;未来正迎面走来。根据世界经济论坛全球议程理事会关于《未来软件与社会》的一份报告&#xff0c;到2025年前&#xff0c;很多新兴技术将会达到其…