BMP文件格式,RGB之间格式转换 碰到坑,MARK

很多人在转储bmp文件的时候,会出现各种各样的问题,特别是抓屏的时候,经常保存下来的图片 怪怪的,偏差很大!比如下面:

 

 有时候,明明感觉自己是对的,但往往结果很让人抓狂。

这种情况一般是对bmp文件格式理解不对,或者没有透彻导致,当然至少是显示出来,所以大部分是对的,只是某些地方出错!

网上也有很多bmp文件格式,但都说得不够透彻,导致实际总要走些弯路。

 

bmp是常见图片格式,使用非常广泛。近期在处理ui库的时候,了解下bmp格式,也发现其中一些坑,记录下。

bmp格式很简单,网上搜索一堆,百科上也有介绍,这里就不重复介绍,

最重要就是文件头,主要是 前54字节,可以用UE看看 bmp:

 

 

注意:

1. 红色框里的内容,是重点

2. 表示数值的,都是低在前,包括后面的像素点。例如,RGB888 实际存放顺序是 BGR888

 

颜色:有几种标准,RGB是常见一种,也非常好理解,所以下面重点是RGB颜色标准。

 

由于4色,黑白色,16色,256色,色差失真太大,除了特殊场合,现在很少用了,因此,我也不怎么去尝试。

对于256色,其实存在画板的概念,可以提升图片效果,降低存储空间,有兴趣可以去了解下

 

重点说下:16位色 24位色

注意区别:16色与16位色,16色就是16种颜色,非常单调;16位色,就是用16bit 来表示颜色,有65536种颜色,比较丰富。

24位色就是24bit来表示颜色,那么有256*256*256种颜色,那么颜色更丰富,满足大部分的正常使用。

        既然24位色这么好,为什么还有用16位色呢?甚至更低色?主要场合不一样,位数越少,表现效果当然越差了,但存储占空间就越少,意味着处理就更快,在显示要求并不是那么高的地方,还是很有市场的,另外颜色也受到显示对象限制,有年代的人应该知道以前屏幕是单色的,因此再多的色彩最终还是 0或1;后来出现彩色屏幕,但颜色也很少,就那么几种,但比单色效果好很多了;再后来出现 真彩色,才使得我们进入888的时代,当然以后更多,颜色也更绚丽。总之,颜色取决于你的设备/处理能力/硬件成本。

 

前面说到的 RGB888就是 24位色=8+8+8

那么16位色又是?RGB565=5+6+5,还有RGB555=5+5+5 ,一般RGB565比较多。

 

“在我们的计算机中图像是以RGB888格式显示图像的,24位图每个像素保存了32bit的数据,即RGB888+Alpha,Alpha就是半透明填充字节……”  但设备当中,我们经常用RGB565,因此两者之间是要转换的,转换算法,网上很多,也很简单,这里就不重复描述。

 

当然实际不会这么简单,我就碰到这么一个坑,通过坑,就好好理解BMP格式。

不知道大家有没有注意到bmp格式里的 001EH 这里的含义: 

  •            0 - 不压缩 (使用BI_RGB表示)
  •       1 - RLE 8-使用8位RLE压缩方式(用BI_RLE8表示)
  •       2 - RLE 4-使用4位RLE压缩方式(用BI_RLE4表示)
  •       3 - Bitfields-位域存放方式(用BI_BITFIELDS表示)   一般用于bit12以及bit32,这里是 重点!重点!重点!重点!

 

因为目标文件是RGB565,16bit,所以 001E这里要填 3,注意不能填0!注意不能填0 !注意不能填0!

那么填了3是否就OK了呢?不是!因为16bit,还有RGB565 以及 RGB555两种标准,你还要告诉对方用哪种格式!不然系统就傻了!

那么怎么告诉系统呢?需要在 原先54字节 后面增加一些信息,主要增加16字节内容:

内容 长度 填写(注意还是要低在前)
红色占位 4字节 RGB565 要填:0x0000F800
绿色占位 4字节 RGB565 要填:0x000007E0
蓝色占位 4字节 RGB565 要填:0x0000001F
预留 4字节 填00

 

注意:

由于头信息变成 70字节了,因此 偏移量/长度  地方的值也要随着改变

还有一个地方:000EH 这个地方要增加16,这里要填 0x38 !

 

     ps:由此可见,如果是RGB555的话,这些地方也要改

可以看下实际51*51的头,用UE:

 

 

还有一个坑,就是bmp要4字节对齐,本来是很好理解的,但实际却是

“每一行都得4字节对齐,不足的用 00H 补齐”

举个例子:

如果图片是 51 * 51  格式: RGB565(一个像素点为2字节)

每行字节数=51*2=102字节,102不是4倍数,所以要补偿两个00,使之=104字节。

实际文件长度一共补偿了102字节。

 

 

 

下面给一个我创建bmp头函数,就可以实现自己的bmp头,由于平时就用到RGB888 跟 RGB565 两种,所以没考虑其他类型的,如果有兴趣可以完善以下代码,使之更加通用!

  1 //pTitle 固定为70字节, 创建
  2 //byBits 位数,比如16 24
  3 //nWidth 图片宽度(多少像素点)
  4 //nHeight 图片高度(多少像素点)
  5 //nFileLen 图片总像素点 = nWidth * nHeight
  6 void CreateBmpTitle(byte *pTitle,byte byBits,UINT nWidth,UINT nHeight,UINT nFileLen)
  7 {
  8     int nPos = 0;
  9     memset(pTitle,0,70);
 10  
 11     //每个像素占多少字节
 12     byte bmpBytes = 2;
 13     if(byBits <= 8)
 14     {
 15         bmpBytes = 1;
 16     }
 17     else if(byBits <= 16)
 18     {
 19         bmpBytes = 2;
 20     }
 21     else if(byBits <= 24)
 22     {
 23         bmpBytes = 3;
 24     }
 25     else
 26     {
 27         bmpBytes = 4;
 28     }
 29  
 30     //每一行都要凑齐4
 31     UINT nLineLen = nWidth * bmpBytes;
 32     if((nLineLen % 4) != 0)
 33     {
 34         UINT nAdd = (4 - (nLineLen % 4)) * nHeight;
 35         nFileLen += nAdd;
 36     }
 37  
 38     pTitle[nPos++] = 'B';
 39     pTitle[nPos++] = 'M';
 40  
 41     //文件长度
 42     SetUINT_L(nFileLen+70,pTitle,nPos);
 43     nPos += 8;//预留4也跳过
 44  
 45     //偏移量
 46     SetUINT_L(70,pTitle,nPos);
 47     nPos += 4;
 48  
 49     //windows, 56
 50     SetUINT_L(56,pTitle,nPos);
 51     nPos += 4;
 52  
 53     //宽度
 54     SetUINT_L(nWidth,pTitle,nPos);
 55     nPos += 4;
 56     //高度
 57     SetUINT_L(nHeight,pTitle,nPos);
 58     nPos += 4;
 59     //平面数=1
 60     SetUSHORT_L(1,pTitle,nPos);
 61     nPos += 2;
 62  
 63     //bits 数
 64     SetUSHORT_L((USHORT)byBits,pTitle,nPos);    
 65     nPos += 2;
 66  
 67     if(byBits == 24)
 68     {
 69         nPos += 4;
 70     }
 71     else if((byBits == 16) || (byBits == 15) || (byBits == 32))
 72     {
 73         //BI_BITFIELDS
 74         SetUINT_L(3,pTitle,nPos);
 75         nPos += 4;
 76     }
 77  
 78     //内容长度,必须为4倍数
 79     SetUINT_L(nFileLen,pTitle,nPos);
 80     nPos += 4;
 81  
 82     SetUINT_L(0xb12,pTitle,nPos);
 83     nPos += 4;
 84  
 85     SetUINT_L(0xb12,pTitle,nPos);
 86     nPos += 4;
 87  
 88     if(byBits == 16)
 89     {
 90         // 565 分别表示 RGB 所在bit位数
 91         nPos = 54;
 92  
 93         SetUINT_L(0xF800,pTitle,nPos);
 94         nPos += 4;
 95  
 96         SetUINT_L(0x07E0,pTitle,nPos);
 97         nPos += 4;
 98  
 99         SetUINT_L(0x001F,pTitle,nPos);
100         nPos += 4;
101     }
102  
103  
104     ///其他都是0
105 }

 

posted @ 2019-10-11 10:04  小刚学长  阅读(2510)  评论(0编辑  收藏  举报