如何使用C++的读取文件流读取一个24位真彩色的BMP文件到一个数组当中

如题所述

BMP文件由文件头、位图信息头、颜色信息和图形数据四部分组成。

一、BMP文件头
BMP文件头数据结构含有BMP文件的类型、文件大小和位图起始位置等信息。其结构定义如下:
typedef struct tagBITMAPFILEHEADER
{
WORD bfType; // 位图文件的类型,必须为BM
DWORD bfSize; // 位图文件的大小,以字节为单位
WORD bfReserved1; // 位图文件保留字,必须为0
WORD bfReserved2; // 位图文件保留字,必须为0
DWORD bfOffBits; // 位图数据的起始位置,以相对于位图文件头的偏移量表示,以字节为单位
} BITMAPFILEHEADER; 二、位图信息头
BMP位图信息头数据用于说明位图的尺寸等信息。其结构定义如下:
typedef struct tagBITMAPINFOHEADER
{
DWORD biSize; // 本结构所占用字节数
LONG biWidth; // 位图的宽度,以像素为单位
LONG biHeight; // 位图的高度,以像素为单位
WORD biPlanes; // 目标设备的级别,必须为1
WORD biBitCount// 每个像素所需的位数,必须是1(双色),4(16色),8(256色)或24(真彩色)之一
DWORD biCompression; // 位图压缩类型,必须是 0(不压缩),1(BI_RLE8压缩类型)或2(BI_RLE4压缩类型)之一
DWORD biSizeImage; // 位图的大小,以字节为单位
LONG biXPelsPerMeter; // 位图水平分辨率,每米像素数
LONG biYPelsPerMeter; // 位图垂直分辨率,每米像素数
DWORD biClrUsed;// 位图实际使用的颜色表中的颜色数
DWORD biClrImportant;// 位图显示过程中重要的颜色数
} BITMAPINFOHEADER; 三、颜色表和位图信息
颜色表用于说明位图中的颜色,有若干个表项,每一个表项是一个RGBQUAD类型的结构,定义一种颜色。RGBQUAD结构的定义如下:
typedef struct tagRGBQUAD
{
BYTErgbBlue;// 蓝色的亮度(值范围为0-255)
BYTErgbGreen; // 绿色的亮度(值范围为0-255)
BYTErgbRed; // 红色的亮度(值范围为0-255)
BYTErgbReserved;// 保留,必须为0
} RGBQUAD; 位图信息头和颜色表组成位图信息,BITMAPINFO结构定义如下:
typedef struct tagBITMAPINFO
{
BITMAPINFOHEADER bmiHeader; // 位图信息头
RGBQUAD bmiColors[1]; // 颜色表
} BITMAPINFO;四、数据读取和颜色分离
Bmp文件有个重要特性,那就是对于数据区域而言,每行的数据它必须凑满4字节,如果没有满,则用冗余的数据来补齐。这个特性直接影响到我们读取位图数据的方法,因为在我们看来(x,y)的数据应该在 y*width+x这样的位置上 但是因为会有冗余信息 那么必须将width用width+该行的冗余量来处理,而由于位图文件有不同的位数,所以这样的计算也不尽相同。

1位:
for(int i=0; i<height; i++)
for(int j=0; j<width; j=j+8)
{
int k=7;
while(k>=0)
{
color[i][k+j]=buffer[n]%2;
buffer[n]=buffer[n]/2;
k--;
}
n++;
}4位: int pitch;
if(width%8==0)
pitch=width;
else
pitch=width+8-width%8; for(int i=0; i<height; i++)
for(int j=0; j<width; j++)
{
int index;
if(j%2==0)
index = buffer[(i*pitch+j)/2]/16;
if(j%2==1)
index = buffer[(i*pitch+j)/2]%16; UCHAR r=quad[index].rgbRed;
UCHAR g=quad[index].rgbGreen;
UCHAR b=quad[index].rgbBlue;
8位: int pitch;
if(width%4==0)
{
pitch=width;
}
else
{
pitch=width+4-width%4;
}
index=buffer[y*pitch+x]; //因为8位位图的数据区域存放的是调色板索引值,所以只需读取这个index

颜色分离:
UCHAR r=quad[index].rgbRed;
UCHAR g=quad[index].rgbGreen;
UCHAR b=quad[index].rgbBlue;

16位:
int pitch=width+width%2;
buffer[(y*pitch+x)*2]
buffer[(y*pitch+x)*2+1]
两个UCHAR内,存放的是(x,y)处的颜色信息
颜色分离:
1.若bitmapinfoheader中的biCompression为BI_RGB时,为555格式,分离代码如下:
UCHAR b=buffer[(i*pitch+j)*2]&0x1F;
UCHAR g=(((buffer[(i*pitch+j)*2+1]<<6)&0xFF)>>3)+(buffer[(i*pitch+j)*2]>>5);
UCHAR r=(buffer[(i*pitch+j)*2+1]<<1)>>3;

2.若bitmapinfoheader中的biCompression为BI_BITFIELDS时,在位图数据区域前存在一个RGB掩码的描述 是3个DWORD值,
我们只需要读取其中的R或者G的掩码,来判断是那种格式。 以红色掩码为例 0111110000000000的时候就是555格式
1111100000000000就是565格式。 565格式分离代码如下:
UCHAR b=buffer[(i*pitch+j)*2]&0x1F;
UCHAR g=(((buffer[(i*pitch+j)*2+1]<<5)&0xFF)>>2)+(buffer[(i*pitch+j)*2]>>5);
UCHAR r=buffer[(i*pitch+j)*2+1]>>3;

24位:
int pitch=width%4;
buffer[(y*width+x)*3+y*pitch];
buffer[(y*width+x)*3+y*pitch+1];
buffer[(y*width+x)*3+y*pitch+2];

颜色分离:
UCHAR b=buffer[(i*width+j)*3+realPitch];
UCHAR g=buffer[(i*width+j)*3+1+realPitch];
UCHAR r=buffer[(i*width+j)*3+2+realPitch];32位:
由于一个象素就是4字节 所以无需补齐

颜色分离:
UCHAR b=buffer[(i*width+j)*4];
UCHAR g=buffer[(i*width+j)*4+1];
UCHAR r=buffer[(i*width+j)*4+2];
温馨提示:内容为网友见解,仅供参考
无其他回答

如何使用C++的读取文件流读取一个24位真彩色的BMP文件到一个数组...
1.读取文件。使用System.IO命名空间加的一个类库。读取一个文件可以按文本的方式读取,也可以按二进制流的方式读取,按二进制流的方式读取,返回的结果就是一个字节数组。byte[] bmps = System.IO.File.ReadAllBytes("a.bmp");2.所谓路径。路径是文件系统的一个概念,对于一个Windows文件系统来说,...

如何使用C++的读取文件流读取一个24位真彩色的BMP文件到一个数组...
} BITMAPINFO;四、数据读取和颜色分离 Bmp文件有个重要特性,那就是对于数据区域而言,每行的数据它必须凑满4字节,如果没有满,则用冗余的数据来补齐。这个特性直接影响到我们读取位图数据的方法,因为在我们看来(x,y)的数据应该在 y*width+x这样的位置上 但是因为会有冗余信息 那么必须将width用w...

用C++完成一个程序,能读取BMP图象文件...
BITMAPFILEHEADER fileHead;fileHead.bfType = 0x4D42;\/\/bmp类型 \/\/bfSize是图像文件4个组成部分之和 fileHead.bfSize= sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER)+ colorTablesize + lineByte*height;fileHead.bfReserved1 = 0;fileHead.bfReserved2 = 0;\/\/bfOffBits是图像文件前3个部分...

摄像机标定:c++怎么读取txt文件命名的jpg格式图片,并且用于摄像机标定...
ifstream fin("calibdata.txt"); \/* 标定所用图像文件的路径 *\/ ofstream fout("caliberation_result.txt"); \/* 保存标定结果的文件 *\/ \/\/读取每一幅图像,从中提取出角点,然后对角点进行亚像素精确化 cout << "开始提取角点………"; int image_count = 0; \/* 图像数量 *\/ Size image_size; \/* ...

如何用C++获取一张bmp图片(24位)的RGB量及长宽
bmWidthBytes + x*bi.bmBitsPixel\/8;int out = RGB(pBit[n+2],pBit[n+1],pBit[n]);\/\/定位到数据中获取颜色信息 GdiplusShutdown(gdiplusToken);\/\/关闭GDI+ } 这里是用GDI+加载位图,也可以用第三方库加载位图获取位图数据然后直接定位像素,使用的时候记得加GDI+的头文件,否则编译不过 ...

关于怎么用C++读取bmp图片
1、图片文件是有固定格式的,像BMP图片是文件头+位图的颜色数据。文件头一般在读取的时候是使用下面的代码:BITMAPFILEHEADER fileheader={0}; fread(&fileheader,sizeof(fileheader),1,fp); if(fileheader.bfType!=0x4D42) \/\/ 判断是否为BMP图片 { fclose(fp); return ; } ...

用C++ 读取bmp信息,求代码
return 1;} void MazeToArray(bmpMaze maze){ \/\/将图片像素信息读取到Ma数组中 int status,i=0,x,y,n=8,j=0,line;line=maze.datasize\/maze.height;for(x=maze.height-1;x>=0;x--){ for(y=0;y<maze.width;y++){ \/\/获取这个点是黑的还是白的 status=maze.bmp[i]&(int)pow(2,...

求助c++读取BMP文件
这时你在运行一下试试。还有,24位位图每个像素用3个字节表示。你的256像素*256像素的图像的实际占用的字节数应是256*256*3 = 196608,这正是biSizeImage的值!bmp文件分为四个部分:文件信息头、位图信息头、调色板、位图信息数据。你在读信息头文件后,应该读调色板数据。对于真彩色图像,因为没有...

怎么用C语言读入、转存一个 BMP图片啊。。。
给你C++的代码吧呵呵,C的也差不多,稍微修改一下就行了 主要是你要了解BMP文件的结构,就是文件头那里所包含的信息,这里用了BITMAPINFOHEADER等现成的结构体来处理,BMP的文件头百度一下就知道的啦,很多资料 include "fstream.h"\/\/24bit bitmap bool CBitmapWindow::LoadFile (char ...

怎么用Visual C++读取BMP格式图像?
Bitmap文件格式...BITMAPFILEHEADER BITMAPINFOHEADER 调色板 位图数据 如果定义高度为正数则数据从下向上存储,否则从上向下。每行的数据要4字节对齐。

相似回答