.net字节流转换为Bitmap图像

本文背景:由于项目用到了C++库进行开发,该库一个回调函数中将位图数据的图像数据作为byte[]传入,用作显示。由于只有图像数据信息,而没有信息头等,所以直接使用Bitmap bitmap = new Bitmap(stream)来构造位图对象时会报参数错误。网上查找资料也未找到相关原因,不过据报错内容推测,应该是数据格式有误。所以考虑到了Win32下位图的格式,想到微软不会因为语言不同而搞两套不同的格式吧,于是就有了此文。

首先看看位图文件的组成部分:
位图文件主要分为如下3个部分:

块名称 对应Windows结构体定义 大小(Byte)
文件信息头 BITMAPFILEHEADER 14
位图信息头 BITMAPINFOHEADER 40
RGB颜色阵列 BYTE* 由图像长宽尺寸决定
一、文件信息头(BITMAPFILEHEADER)的结构如下:

typedef struct tagBITMAPFILEHEADER {
        WORD    bfType;
        DWORD   bfSize;
        WORD    bfReserved1;
        WORD    bfReserved2;
        DWORD   bfOffBits;
} BITMAPFILEHEADER, FAR *LPBITMAPFILEHEADER, *PBITMAPFILEHEADER;

bfType 说明文件的类型,该值必需是0x4D42,也就是字符’BM’。
bfSize 说明该位图文件的大小,用字节为单位
bfReserved1 保留,必须设置为0
bfReserved2 保留,必须设置为0
bfOffBits 说明从文件头开始到实际的图象数据之间的字节的偏移量。这个参数是非常有用的,因为位图信息头和调色板的长度会根据不同情况而变化,所以你可以用这个偏移值迅速的从文件中读取到位数据。一般情况下大小为文件头加上位图信息头,即14+40=54

二、位图信息头(BITMAPINFOHEADER)结构如下:

typedef struct tagBITMAPINFOHEADER{
        DWORD      biSize;
        LONG       biWidth;
        LONG       biHeight;
        WORD       biPlanes;
        WORD       biBitCount;
        DWORD      biCompression;
        DWORD      biSizeImage;
        LONG       biXPelsPerMeter;
        LONG       biYPelsPerMeter;
        DWORD      biClrUsed;
        DWORD      biClrImportant;
} BITMAPINFOHEADER, FAR *LPBITMAPINFOHEADER, *PBITMAPINFOHEADER;

biSize 说明BITMAPINFOHEADER结构所需要的字数。
biWidth 说明图象的宽度,以象素为单位。
biHeight 说明图象的高度,以象素为单位。注:这个值除了用于描述图像的高度之外,它还有另一个用处,就是指明该图像是倒向的位图,还是正向的位图。如果该值是一个正数,说明图像是倒向的,如果该值是一个负数,则说明图像是正向的。大多数的BMP文件都是倒向的位图,也就是时,高度值是一个正数。
biPlanes 为目标设备说明位面数,其值将总是被设为1。
biBitCount 说明比特数/象素,其值为1、4、8、16、24、或32。但是由于我们平时用到的图像绝大部分是24位和32位的,所以我们讨论这两类图像。
biCompression 说明图象数据压缩的类型,同样我们只讨论没有压缩的类型:BI_RGB。
biSizeImage 说明图象的大小,以字节为单位。当用BI_RGB格式时,可设置为0。
biXPelsPerMeter 说明水平分辨率,用象素/米表示。
biYPelsPerMeter 说明垂直分辨率,用象素/米表示。
biClrUsed 说明位图实际使用的彩色表中的颜色索引数(设为0的话,则说明使用所有调色板项)。
biClrImportant 说明对图象显示有重要影响的颜色索引的数目,如果是0,表示都重要。
三、位图数据
1、RGB颜色阵列
有关RGB三色空间我想大家都很熟悉,这里我想说的是在Windows下,RGB颜色阵列存储的格式其实BGR。也就是说,对于24位的RGB位图像素数据格式是:
蓝色B值 绿色G值 红色R值
对于32位的RGB位图像素数据格式是:
蓝色B值 绿色G值 红色R值 透明通道A值
透明通道也称Alpha通道,该值是该像素点的透明属性,取值在0(全透明)到255(不透明)之间。对于24位的图像来说,因为没有Alpha通道,故整个图像都不透明。
2、行对齐
由于Windows在进行行扫描的时候最小的单位为4个字节,所以当图片宽 X 每个像素的字节数 != 4的整数倍时,要在每行的后面补上缺少的字节,以0填充(一般来说当图像宽度为2的幂时不需要对齐)。位图文件里的数据在写入的时候已经进行了行对齐,也就是说加载的时候不需要再做行对齐。但是这样一来图片数据的长度就不是:宽 X 高 X 每个像素的字节数 了,我们需要通过下面的方法计算正确的数据长度:
//Calculate the image data size
int LineByteCnt = (((ImageWidth * biBitCount) + 31) >> 5) << 2;
ImageDataSize = LineByteCnt * ImageHeight;

知道了位图的结构以及要求后,手动填充信息头部分完成到Bitmap对象的转换。

public System.Windows.Media.Imaging.BitmapImage GetBitmapFromMemory(byte[] imageData,int length)
        {
            Bitmap bitmap = null;
//             int length = imagedatadetails.Length;
            using (MemoryStream stream = new MemoryStream(length*4 + 14 + 40))//为头腾出54个长度的空间
            {
                //开始写文件信息头
                byte[] buffer = new byte[13];
                buffer[0] = 0x42;//bitmap 固定常数
                buffer[1] = 0x4d;//bitmap 固定常数
                stream.Write(buffer, 0, 2);//先写入头的前两个字节
                //把我们之前获得的数据流的长度转换成字节,
                //这个是用来告诉“头”我们的实际图像数据有多大
                byte[] bytes = BitConverter.GetBytes(length*4);
                stream.Write(bytes, 0, 4);//把这个长度写入头中去

                bytes = BitConverter.GetBytes(0)
                stream.Write(buffer, 0, 4);//在写入4个字节长度的数据到头中去

                int num2 = 54;//bitmap 固定常数也就是十六进制的0x36
                bytes = BitConverter.GetBytes(num2);
                stream.Write(bytes, 0, 4);//在写入最后4个字节的长度

                //开始写入位图信息头
                bytes = BitConverter.GetBytes(40);  //写入信息头的长度biSize
                stream.Write(bytes, 0, 4);
                bytes = BitConverter.GetBytes(118);  //写入信息头的图像宽度biWidth
                stream.Write(bytes, 0, 4);
                bytes = BitConverter.GetBytes(68);  //写入信息头的图像高度biHeight
                stream.Write(bytes, 0, 4);

                bytes = BitConverter.GetBytes((short)1);  //写入信息头的biPlanes
                stream.Write(bytes, 0, 2);

                bytes = BitConverter.GetBytes((short)32);  //写入信息头的biBitCount
                stream.Write(bytes, 0, 2);

                bytes = BitConverter.GetBytes(0);  //写入信息头的biCompression
                stream.Write(bytes, 0, 4);

                bytes = BitConverter.GetBytes(0);  //写入信息头的biSizeImage
                stream.Write(bytes, 0, 4);

                bytes = BitConverter.GetBytes(0);  //写入信息头的biXPelsPerMeter
                stream.Write(bytes, 0, 4);

                bytes = BitConverter.GetBytes(0);  //写入信息头的biYPelsPerMeter
                stream.Write(bytes, 0, 4);

                bytes = BitConverter.GetBytes(0);  //写入信息头的biClrUsed
                stream.Write(bytes, 0, 4);

                bytes = BitConverter.GetBytes(0);  //写入信息头的biClrImportant
                stream.Write(bytes, 0, 4);

                //stream.GetBuffer();
//写入位图数据,由于我的数据是灰度单通道图像,所以复制4个字节数据作为一个RGBA的像素
                for (int i = 0; i < length; ++i)
                {
                    byte[] x = new byte[] {imageData[i]};
                    stream.Write(x, 0, 1);
                    stream.Write(x, 0, 1);
                    stream.Write(x, 0, 1);
                    stream.Write(x, 0, 1);
                }

                bitmap = new Bitmap(stream);//用内存流构造出一幅bitmap的图片
                bitmap.RotateFlip(RotateFlipType.Rotate180FlipX);

                MemoryStream memstream = new MemoryStream();
                bitmap.Save(memstream,System.Drawing.Imaging.ImageFormat.Bmp);
                BitmapImage image = new BitmapImage();
                image.BeginInit();
                image.StreamSource = memstream;
                image.EndInit();
                stream.Close();
                bitmap.Dispose();

                return image;//最后就得到了我们想要的图片了
            }
        }

如上函数将内存中的字节流转换为了c#中可以使用的Bitmap或BitmapImage对象,可以直接贴到窗口上进行显示。

  • 5
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值