HBITMAP与BITMAP 的区别 BMP图像的格式

来源:互联网 发布:spark 网页端口 编辑:程序博客网 时间:2024/06/05 06:12

HBITMAP   是句柄;  
BITMAP    是实例:  

typedef struct tagBITMAP {  /* bm */
    int     bmType;//必须是BM
    int     bmWidth;//指定位图的宽度(以象素为单位)
    int     bmHeight;//指定位图的高度(以象素为单位)。
    int     bmWidthBytes;//一行几个字节,4位对齐
    BYTE    bmPlanes;//指定目标设备的位面数
    BYTE    bmBitsPixel;//指定每个象素的位数
    LPVOID  bmBits;//指向图象数据的指针

} BITMAP;

注意 bmBits是自己先前设置的  具体看下面复制的:

CBitmap::CreateBitmapIndirect函数的功能是用一个BITMAP结构体重的高度、宽度和位模式(如果指定了一个的话)来初始化一个位图。调用该函数时,用户可以设置bmBits字段为NULL或者设为像素位数据的地址(用以初始化该位图)。
复制自:http://bbs.csdn.net/topics/10310605

所以如果我们生成位图的时候没有设置bmBits  ,之后通过下面的 方法得到的bm结构中的bmBits是NULL   是无效的指针   

这时候我们可以通过GetDIBits获得bmp图像的DIB格式的数据.  我们传入的第6个参数 即LPBITMAPINFO  结构体会接收返回的位图信息 比如位图数据大小     具体见下面BITMAPINFOHEADER的解释 

    

windows不允许直接对用户对象操作,其操作得通过句柄来进行。  

例如:
  HBITMAP   hBmp;  
  BITMAP   bm;  
  GetObject(hBmp,sizeof(bm),&bm);  

通过HBITMAP取得BITMAP;

以下不是转载  而是个人理解  不对请指正:注意上面得到的bm是hBmp的信息  它表示的是hBmp的固有属性  假如它是与你的屏慕兼容的bmp   那么hBmp的信息记录的是屏幕位图的信息 不会改变。

GetDIBits 当你用位图去获取信息时,获取信息的位数 什么的 跟你传入的BITMAPINFO结构有关系 假如这个bmp已经被选入屏幕DC  你要获得屏幕的8位图像(256色) 而一般你的屏幕是32位  那么bm中存放的 bmBitsPixel是32   而你想到8位的话 就 需要在BITMAPINFO结构中设置 然后给GetDIBits作参数!

 



BMP图像   以下是有部分信息是摘抄自其它网友的   

由四部分组成:

A. 第一部分为位图文件头BITMAPFILEHEADER,它是一个结构体,定义如下:

typedef struct tagBITMAPFILEHEADER{

     WORD    bfType; /文件类型,必须是0x424D,即字符串“BM”

     DWORD   bfSize; /指定文件大小,包括这14个字节

     WORD    bfReserved1; /保留字,不用考虑

     WORD    bfReserved2; /保留字,不用考虑

     DWORD   bfOffBits; /从文件头到实际位图数据的偏移字节数

}BITMAPFILEHEADER;

这个结构的长度是固定的,为14个字节(WORD为无符号16位二进制整数,DWORD为无符号32位二进制整数)。

B. 第二部分为位图信息头BITMAPINFOHEADER,也是一个结构,定义如下:

typedef struct tagBITMAPINFOHEADER{

DWORD   biSize; /该结构的长度,为40

LONG      biWidth; / 图像的宽度,单位是像素

LONG      biHeight; / 图像的高度,单位是像素

WORD      biPlanes; / 位平面数,必须是1,不用考虑

WORD      biBitCount;/ 指定颜色位数,1为二值,4为16色,8为256色,16、24、32为真彩色 DWORD     biCompression; / 指定是否压缩,有效的值为BI_RGB,BI_RLE8,BI_RLE4,BI_BITFIELDS

DWORD     biSizeImage; / 实际的位图数据占用的字节数

LONG      biXPelsPerMeter; / 目标设备水平分辨率,单位是每米的像素数

LONG      biYPelsPerMeter; / 目标设备垂直分辨率,单位是每米的像素数

DWORD     biClrUsed;/实际使用的颜色数,若该值为0,则使用颜色数为2的bitBitCount次方种

DWORD     biClrImportant; /图像中重要的颜色数,若该值为0,则所有的颜色都是重要的

} BITMAPINFOHEADER;

这个结构的长度是固定的,为40个字节(LONG为32为二进制整数)。

偏移域的名称长度/字节内容000Eh(14)biSize4
文件信息头的长度
40(28h):Windows
00012h(18)biWidth4位图的宽度,以像素为单位00016h(22)biHeight4位图的高度,以像素为点为001Ah(26)Biplanes2位图的为平面数(该值总是为1)001Ch(28)biBitCount2
每个像素的位平面数,有下面几种情况:
1:单色位图
4:16色位图
8:256色位图
16:16bit高彩色位图
24:24bit真彩色位图
32:32bit增强型真彩色位图
001Eh(30)biCompression4
压缩说明:
0:不压缩(用BI_RGB表示)
1:RLE8,使用8位RLE压缩方式(用BI_RLE8表示)
2:RLE4,使用4位RLE压缩方式(用BI_RLE4表示)
3:Bitfields:位域存放方式(用BI_BITFIELDS表示)
0022h(34)biSizeImage4位图数据的大小,以字节为单位。该数必须是4的倍数。当图像存储的是非压缩数据的时候,它的取值可以为0,实际上,此时位图数据的大小可以通过biBitCount,biWidth,biHeight等计算出来0026h(38)biXPelsPerMeter4用像素/米表示的水平分辨率002Ah(42)biYPelsPerMeter4用像素/米表示的垂直分辨率002Eh(46)biClrUsed4位图使用的颜色数,如果为0则表示使用了全部可能的颜色0032h(50)biClrImportant4指定重要的颜色数。当该域的值等于颜色数时(或者等于0时),表示所有颜色都一样重要

C. 第三部分为调色板(Palette),当然,这里是对那些需要调色板的位图文件而言的。真彩色图像不需要调色板,BITMAPINFOHEADER后直接是位图数据。调色板实际上是一个数组,共有biClrUsed个元素(如果该值为零,则有2的biBitCount次方个元素)。数组中每个元素的类型是一个RGBQUAD结构,占4个字节,其定义如下:

    typedef struct tagRGBQUAD

{

       BYTE rgbBlue;   //该颜色的蓝色分量

       BYTE rgbGreen;   //该颜色的绿色分量

       BYTE rgbRed;     //该颜色的红色分量

       BYTE rgbReserved;//保留值

} RGBQUAD;

D.第四部分是实际的图像数据,对于用到调色板的位图,图像数据就是该像素颜色在调色板中的索引值,对于真彩色图像,图像数据就是实际的R、G、B值:

• 对于2色位图:用1位就可以表示该像素的颜色(一般0表示黑,1表示白),所以一个

字节就可以表示8个像素;

• 对于16色位图,用4位可以表示一个像素的颜色,所以一个字节可以表示2个像素;

• 对于256色位图,一个字节刚好可以表示1个像素;

色深24位的位图

★     每一行的字节数必须是4的整数倍,如果不是,则需要补齐;(bitSizeImage中提到过)

★     BMP文件的数据存放是从下到上,从左到右的。也就是说,从文件中最先读到的是图像最下面的左边第一个像素,然后是左边第二个像素,接下来是倒数第二行左边第一个像素,左边第二个像素,依次类推

每个像素用3个字节表示,顺序依次为红,绿和蓝的值。每行用0填充到4字节的边界。



读取图像数据的方法:

HBITMAP hBmp = (HBITMAP)LoadImage( NULL , "z:\\bg4.bmp" , IMAGE_BITMAP , 0 , 0 , LR_CREATEDIBSECTION | LR_LOADFROMFILE); if (hBmp != NULL) {  BITMAP bmp = {0};  int ret = GetObject( hBmp , sizeof(BITMAP) , &bmp );  if (ret)  {   TRACE( "Width = %d Height = %d BitsPerPixel = %d\n" , bmp.bmWidth , bmp.bmHeight , bmp.bmBitsPixel );   BYTE* pPixel = (BYTE*)bmp.bmBits;  //指向BMP像素的指针   int x = 1439 , y = 899;  //指定要获取像素的坐标  规定图片左上角坐标为0,0   BYTE *pOffset = pPixel + (bmp.bmHeight - 1) * bmp.bmWidthBytes;  //指向最后一行像素数据   pOffset =  (pOffset -  y * bmp.bmWidthBytes) + x * bmp.bmBitsPixel / 8; //指向指定像素位置   DWORD rgb = 0;   memcpy( &rgb , pOffset , bmp.bmBitsPixel / 8 );   TRACE( "(x = %d y = %d) = %u\n" , x , y , rgb );  }    DeleteObject( hBmp );  hBmp = NULL;  ZeroMemory( &bmp , sizeof(bmp) ); }//上面代码里的rgb就是你要的像素了//如果你是在对话框之类的处理的话可以用类似下面的代码实现CDC* pDc = GetDC(); POINT pt; pt.x = 10; pt.y = 10; pDc->SetPixel( pt , RGB(255,0,0) );



部分摘抄来自http://blog.sina.com.cn/s/blog_48f0f4da0100c3es.html

      http://blog.sina.com.cn/s/blog_49dd59fe01011wbj.html

0 0
原创粉丝点击