关于怎么用C++读取bmp图片

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了关于怎么用C++读取bmp图片相关的知识,希望对你有一定的参考价值。

#include "fstream.h"//24bit bitmap
bool CBitmapWindow::LoadFile (char *bmpFile)

FILE *fp;
if((fp=fopen(bmpFile,"rb"))==NULL)
return false;

WORD bfType;
DWORD bfSize;
WORD bfReserved1;
WORD bfReserved2;
DWORD bfOffBits;
BITMAPINFOHEADER bih;
fread(&bfType,sizeof(WORD),1,fp);
if(bfType!=0x4d42)
return false;
fread(&bfSize,sizeof(DWORD),1,fp);
fread(&bfReserved1,sizeof(WORD),1,fp);
fread(&bfReserved2,sizeof(WORD),1,fp);
fread(&bfOffBits,sizeof(DWORD),1,fp);
fread(&bih,sizeof(BITMAPINFOHEADER1),1,fp);
nWidth=bih.biWidth ;
nHeight=bih.biHeight;
if(nWidth % 4 !=0)
nWidth=nWidth+ (4-nWidth % 4); //修正位图宽度值
DWORD size=nWidth*bih.biBitCount/8*nHeight;
arrayColor=new ZafLogicalColor[nWidth*nHeight];
// LPSTR pData=new char[size];
unsigned char* pData=new unsigned char[size];
fread(pData,size,1,fp);

int bmWidthBytes=nWidth*bih.biBitCount /8;
int bmBitsPixel=bih.biBitCount ;
int nBit=bmBitsPixel/8;

colorTableBitmap* clrTableBmp;
clrTableBmp=new colorTableBitmap[nWidth*nHeight];

char cBit[40];
int z;
for (int y=0;y<nHeight;y++)

for (int x=0;x<nWidth;x++)

z=nHeight-y-1;
clrTableBmp[y*nWidth+x].nIndex =y*nWidth+x;
clrTableBmp[y*nWidth+x].iBlue =pData[x*nBit+z*bmWidthBytes];
clrTableBmp[y*nWidth+x].iGreen =pData[x*nBit+1+z*bmWidthBytes];
clrTableBmp[y*nWidth+x].iRed=pData[x*nBit+2+z*bmWidthBytes];
sprintf(cBit,",%d:%d:%d:%d",y*nWidth+x,pData[x*nBit+z*bmWidthBytes],
pData[x*nBit+1+z*bmWidthBytes],
pData[x*nBit+2+z*bmWidthBytes]);



delete []clrTableBmp;
delete []pData;
fclose(fp);
return true;
网上代码是这样的。。可是我不明白,不知道怎么关联我的本地已有的图片,希望您能搞教诲下。。谢谢,不甚感激

可以看下如下的操作:

 1、图片文件是有固定格式的,像BMP图片是文件头+位图的颜色数据。

文件头一般在读取的时候是使用下面的代码:

 BITMAPFILEHEADER fileheader=0; 
    fread(&fileheader,sizeof(fileheader),1,fp); 
    if(fileheader.bfType!=0x4D42)  // 判断是否为BMP图片
     
        fclose(fp); 
        return ; 
     
 
    BITMAPINFOHEADER head; 
    fread(&head,sizeof(BITMAPINFOHEADER),1,fp);  
    long bmpWidth = head.biWidth;     //获取图片的宽
    long bmpHeight = head.biHeight;   //获取图片的宽
    WORD biBitCount = head.biBitCount; 
    if(biBitCount != 24) // 是否为24位位图
     
        ::AfxMessageBox(_T("请选择24位位图!")); 
        fclose(fp); 
        return ; 
    

2、文件读到这里了就要读到位图的颜色数据了,那么在读取前,需要知道数据的大小,这样才能把分配具体大学的缓冲区,之后把数据完整的读取到缓冲区内。

那么计算位图数据的大小一般用下面的代码去做。

int totalSize = (bmpWidth *biBitCount/8+3)/4*4*bmpHeight; 
BYTE *pBmpBuf = new BYTE[totalSize];

3、在这些工作做完之后,就使用fread命令直接读取文件内容即可,直到读到文件结束。

整个具体的代码实现如下:

 FILE *fp=NULL; 
    int ret = fopen_s(&fp,"D:\\\\11.bmp","rb"); 
    if(fp==0)    
     
        return ; 
     
    BITMAPFILEHEADER fileheader=0; 
    fread(&fileheader,sizeof(fileheader),1,fp); 
    if(fileheader.bfType!=0x4D42) 
     
        fclose(fp); 
        return ; 
     
 
    BITMAPINFOHEADER head; 
    fread(&head,sizeof(BITMAPINFOHEADER),1,fp);  
    long bmpWidth = head.biWidth; 
    long bmpHeight = head.biHeight; 
    WORD biBitCount = head.biBitCount; 
    if(biBitCount != 24) 
     
        ::AfxMessageBox(_T("请选择24位位图!")); 
        fclose(fp); 
        return ; 
     
 
    int totalSize = (bmpWidth *biBitCount/8+3)/4*4*bmpHeight; 
    BYTE *pBmpBuf = new BYTE[totalSize]; 
    size_t size = 0; 
    while(true) 
     
        int iret = fread(&pBmpBuf[size],1,1,fp); 
        if(iret == 0) 
            break; 
        size = size + iret; 
     
    fclose(fp); 
 
    int i,j; 
    CClientDC dc(this); 
    int pitch=bmpWidth%4; 
    for(i=0;i<bmpHeight;i++) 
     
        int realPitch=i*pitch; 
        for(j=0;j<bmpWidth;j++) 
         
            dc.SetPixel(j,i,RGB( 
                pBmpBuf[(i*bmpWidth+j)*3+2+realPitch], 
                pBmpBuf[(i*bmpWidth+j)*3+1+realPitch], 
                pBmpBuf[(i*bmpWidth+j)*3+realPitch])); 
         
     
delete [] pBmpBuf;     pBmpBuf = NULL;
    return ;
参考技术A 你好,我把问题补充一下,希望对你有帮助。如果还是不行,你再问我。

在你如下的程序段中:
//读取头文件
fseek( rfile , sizeof(short) ,SEEK_SET );
fread( pointfh ,sizeof(BITMAPFILEHEADER) , 1 , rfile );
//读取信息头文件
fseek( rfile , sizeof(BITMAPFILEHEADER)+sizeof(short) ,SEEK_SET );
fread( pointih ,sizeof(BITMAPINFOHEADER),1, rfile );
//读位图信息
pointimage=(char*)calloc(ih.biSizeImage*3, sizeof(char));
fseek(rfile,fh.bfOffBits,SEEK_SET);//头指针移动到位图区
fread(pointimage,sizeof(char),ih.biSizeImage*3,rfile);

第一句中,我觉得不要加sizeof(short),而是将其改为0。因为文件的前sizeof(BITMAPFILEHEADER)个字节就是 文件信息头,紧接其后的sizeof(BITMAPINFOHEADER)个字节是位图信息头,所以在你的第三条语句中,也应去掉sizeof(short)。另外,biSizeImage就是位图像素信息实际占用的存储空间长度,不需要乘以3,所以在上面第5和第7句中应将“*3”去掉。这时你在运行一下试试。

还有,24位位图每个像素用3个字节表示。你的256像素*256像素的图像的实际占用的字节数应是256*256*3 = 196608,这正是biSizeImage的值!

bmp文件分为四个部分:
文件信息头、位图信息头、调色板、位图信息数据。

你在读信息头文件后,应该读调色板数据。对于真彩色图像,因为没有调色板,所以不用读;但对于其他的图像就需要读了。这是你的代码存在的问题之一。解决办法是根据biBitCount的值判断调色板的大小:若biBitCount为1,调色板大小为2*sizeof(RGBQUAD)字节;若为4,调色板16*sizeof(RGBQUAD);若为8,调色板256*sizeof(RGBQUAD);若为24,调色板为0。

另外,位图信息数据每行存储的字节数一定是4字节的整数倍,不足的时候补0。比如你的图像每行只有一个像素,一个像素用三个字节存储,那么在实际存储该图像时需要在每行多补充一个字节,以使其达到4字节的整数倍。biSizeImage的值就是图像信息数据实际存储的字节数,读的时候直接读这么多字节的数据就行了,不需要乘以3。你的代码在读信图信息时并没考虑这一点,这是存在的问题之二。图像每行存储的实际字节数:nBytesPerLine = biSizeImage / biHeight,这个值在后续的处理中可能会用到。

你说读取的分辨率是3780,是指每米有3780个像素,单位是像素/米。而那个96的单位可能不是像素/米。

希望对你有帮助。

读bmp图片:头文件为66字节。。。怎么用c语言来读取数据啊,网上的代码我看不懂。新手,希望能写明白些

网上的程序太长且无重点,希望能将关键性语句注释。

参考技术A /* 单色BMP图形文件显示 */

#include "graphics.h"
#include<stdio.h>
#include<conio.h>
/* BMP 文件头部结构 */
struct header

unsigned int bfType;
long bfSize;
unsigned int bfReserved1;
unsigned int bfReserved2;
long bfoffBits;
;

struct tinfoheader

long biSize;
long biWidth;
long biHeight;
unsigned int biPlanes;
unsigned int biBitCount;
long biCompression;
long biSizeImage;
long biXPelsPerMeter;
long biYPelsPerMeter;
long biClrUsed;
long biClrImportant;
;
struct header HEADER;
struct tinfoheader INFO;

unsigned char dgbuff[10000];/* 黑白图形文件大小,不能超过 */

void LoadDZ()

int i,ld;
char filename[20]="1.bmp";/*假设在当前目录下1.bmp文件 */
FILE *fp;

fp=fopen(filename,"rb");
fseek(fp,0,SEEK_SET);
fseek(fp,0,SEEK_END);
ld=ftell(fp);
fclose(fp);
fp=fopen(filename,"rb");
if(fp!=NULL)

fread(&dgbuff,ld,1,fp);
fclose(fp);

fp=fopen(filename,"rt");
if(fp!=NULL)

fread(&HEADER, sizeof(struct header), 1, fp);
fread(&INFO, sizeof(struct tinfoheader), 1, fp);
fclose(fp);



void show(int x,int y,int giColor,int backColor)/* 显示位置、前景色、背景色 */

int i,j,k,l=0,num,m,x0,x1,x2;
unsigned char b,c,c1;

if(HEADER.bfType!=0x4d42 || HEADER.bfReserved1!=0 || INFO.biSizeImage==0
|| INFO.biPlanes!=1 || INFO.biBitCount!=1 || INFO.biCompression!=0)
return;

m=INFO.biWidth/32;
if((INFO.biWidth%32)!=0) m=m+1;
num=HEADER.bfoffBits;

for(l=INFO.biHeight-1; l>=0; l--)

for (i=0;i<m;i++)

x0=i<<5;
for (j=0;j<4;j++)

x1=j<<3;
c=dgbuff[num];
num++;
for (k=7;k>=0;k--)

x2=x0+x1+7-k;
if(x2>=INFO.biWidth) continue;
else

b=1<<k;
c1=c&b;
if(c1==0)

putpixel(x+x2,y+l,giColor);

else
putpixel(x+x2,y+l,backColor);







void main()

int driver,mode;

driver=VGA;mode=VGAHI;
initgraph(&driver,&mode,"");
setfillstyle(SOLID_FILL,WHITE);
bar(0,0,639,479);
LoadDZ();
show(200,200,RED,BLACK);
getch();
closegraph();
追问

如果您能每句话都写个注释,。。。我就把分给你了。谢!

以上是关于关于怎么用C++读取bmp图片的主要内容,如果未能解决你的问题,请参考以下文章

用c ++读取BMP文件的所有字节并旋转图片

base64转化为bmp图片?

用C语言读取16位bmp图片的每个像素的信息~

纯C++实现24位bmp格式图片的读取和修饰

用c语言读取24位位图bmp文件

读bmp图片:头文件为66字节。。。怎么用c语言来读取数据啊,网上的代码我看不懂。新手,希望能写明白些