C语言编写的bmp读写程序

C语言编写的bmp读写程序 建议先把bmp的数据存储格式了解下 

[cpp] view plaincopy

 

  1. <span style="font-size:16px;">#include "Windows.h"  
  2. #include "stdio.h"  
  3. #include "string.h"  
  4. #include "malloc.h"  
  5.   
  6. unsigned char *pBmpBuf;//读入图像数据的指针  
  7. int bmpWidth;//图像的宽  
  8. int bmpHeight;//图像的高  
  9. RGBQUAD *pColorTable;//颜色表指针  
  10. int biBitCount;//图像类型,每像素位数  
  11.   
  12. bool readBmp(char *bmpName)  
  13. {  
  14.   //二进制读方式打开指定的图像文件  
  15.   FILE *fp=fopen(bmpName,"rb");  
  16.   if(fp==0) return 0;  
  17.       
  18.       
  19.   //跳过位图文件头结构BITMAPFILEHEADER  
  20.   fseek(fp, sizeof(BITMAPFILEHEADER),0);  
  21.   //定义位图信息头结构变量,读取位图信息头进内存,存放在变量head中  
  22.   BITMAPINFOHEADER head;     
  23.   fread(&head, sizeof(BITMAPINFOHEADER), 1,fp);    
  24.   //获取图像宽、高、每像素所占位数等信息  
  25.   bmpWidth = head.biWidth;  
  26.   bmpHeight = head.biHeight;  
  27.   biBitCount = head.biBitCount;  
  28.   //定义变量,计算图像每行像素所占的字节数(必须是4的倍数)  
  29.   int lineByte=(bmpWidth * biBitCount/8+3)/4*4;  
  30.   //灰度图像有颜色表,且颜色表表项为256  
  31.   if(biBitCount==8){  
  32.   //申请颜色表所需要的空间,读颜色表进内存  
  33.   pColorTable=new RGBQUAD[256];  
  34.   fread(pColorTable,sizeof(RGBQUAD),256,fp);  
  35.   }  
  36.   //申请位图数据所需要的空间,读位图数据进内存  
  37.   pBmpBuf=new unsigned char[lineByte * bmpHeight];  
  38.   fread(pBmpBuf,1,lineByte * bmpHeight,fp);  
  39.   //关闭文件  
  40.   fclose(fp);  
  41.   return 1;  
  42. }  
  43.   
  44. bool saveBmp(char *bmpName, unsigned char *imgBuf, int width, int height,    
  45.   int biBitCount, RGBQUAD *pColorTable)  
  46. {  
  47.   //如果位图数据指针为0,则没有数据传入,函数返回  
  48.   if(!imgBuf)  
  49.   return 0;  
  50.   //颜色表大小,以字节为单位,灰度图像颜色表为1024字节,彩色图像颜色表大小为0  
  51.   int colorTablesize=0;  
  52.   if(biBitCount==8)  
  53.   colorTablesize=1024;  
  54.   //待存储图像数据每行字节数为4的倍数  
  55.   int lineByte=(width * biBitCount/8+3)/4*4;  
  56.   //以二进制写的方式打开文件  
  57.   FILE *fp=fopen(bmpName,"wb");  
  58.   if(fp==0) return 0;  
  59.   //申请位图文件头结构变量,填写文件头信息  
  60.   BITMAPFILEHEADER fileHead;  
  61.   fileHead.bfType = 0x4D42;//bmp类型  
  62.   //bfSize是图像文件4个组成部分之和  
  63.   fileHead.bfSize= sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER)  
  64.   + colorTablesize + lineByte*height;  
  65.   fileHead.bfReserved1 = 0;  
  66.   fileHead.bfReserved2 = 0;  
  67.   //bfOffBits是图像文件前3个部分所需空间之和  
  68.   fileHead.bfOffBits=54+colorTablesize;  
  69.   //写文件头进文件  
  70.   fwrite(&fileHead, sizeof(BITMAPFILEHEADER),1, fp);  
  71.   //申请位图信息头结构变量,填写信息头信息  
  72.   BITMAPINFOHEADER head;    
  73.   head.biBitCount=biBitCount;  
  74.   head.biClrImportant=0;  
  75.   head.biClrUsed=0;  
  76.   head.biCompression=0;  
  77.   head.biHeight=height;  
  78.   head.biPlanes=1;  
  79.   head.biSize=40;  
  80.   head.biSizeImage=lineByte*height;  
  81.   head.biWidth=width;  
  82.   head.biXPelsPerMeter=0;  
  83.   head.biYPelsPerMeter=0;  
  84.   //写位图信息头进内存  
  85.   fwrite(&head, sizeof(BITMAPINFOHEADER),1, fp);  
  86.   //如果灰度图像,有颜色表,写入文件    
  87.   if(biBitCount==8)  
  88.   fwrite(pColorTable, sizeof(RGBQUAD),256, fp);  
  89.   //写位图数据进文件  
  90.   fwrite(imgBuf, height*lineByte, 1, fp);  
  91.   //关闭文件  
  92.   fclose(fp);  
  93.   return 1;  
  94. }  
  95.   
  96. void main()  
  97. {  
  98.   char inFileName[90],outFileName[90];  
  99. printf("请输入原始位图文件的文件名:");  
  100.   scanf("%s",inFileName);  
  101.   printf("请输入加密程序产生的新位图文件的文件名:");  
  102.   scanf("%s",outFileName);  
  103. //读入指定BMP文件进内存  
  104.   readBmp(inFileName);  
  105.   //输出图像的信息  
  106.   printf("width=%d,height=%d, biBitCount=%d\n",bmpWidth,bmpHeight, biBitCount);  
  107.   //将图像数据存盘  
  108.   saveBmp(outFileName, pBmpBuf, bmpWidth, bmpHeight, biBitCount, pColorTable);  
  109.   //清除缓冲区,pBmpBuf和pColorTable是全局变量,在文件读入时申请的空间  
  110.   delete []pBmpBuf;  
  111.   if(biBitCount==8)  
  112.   delete []pColorTable;  
  113. }  
  114. </span>  

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

一、BMP文件结构 

BMP文件组成 
BMP文件由文件头、位图信息头、颜色信息和图形数据四部分组成。   
BMP文件头 
BMP文件头数据结构含有BMP文件的类型、文件大小和位图起始位置等信息。   
其结构定义如下: 
typedef   struct   tagBITMAPFILEHEADER 

WORDbfType;   //   位图文件的类型,必须为BM 
DWORD   bfSize;   //   位图文件的大小,以字节为单位   
WORDbfReserved1;   //   位图文件保留字,必须为0 
WORDbfReserved2;   //   位图文件保留字,必须为0 
DWORD   bfOffBits;   //   位图数据的起始位置,以相对于位图 
//   文件头的偏移量表示,以字节为单位 
}   BITMAPFILEHEADER; 

3.   位图信息头 

BMP位图信息头数据用于说明位图的尺寸等信息。 
typedef   struct   tagBITMAPINFOHEADER{ 
DWORD   biSize;   //   本结构所占用字节数 
LONGbiWidth;   //   位图的宽度,以像素为单位 
LONGbiHeight;   //   位图的高度,以像素为单位 
WORD   biPlanes;   //   目标设备的级别,必须为1 
WORD   biBitCount//   每个像素所需的位数,必须是1(双色), 
//   4(16色),8(256色)或24(真彩色)之一 
DWORD   biCompression;   //   位图压缩类型,必须是   0(不压缩), 
//   1(BI_RLE8压缩类型)或2(BI_RLE4压缩类型)之一 
DWORD   biSizeImage;   //   位图的大小,以字节为单位 
LONGbiXPelsPerMeter;   //   位图水平分辨率,每米像素数 
LONGbiYPelsPerMeter;   //   位图垂直分辨率,每米像素数 
DWORD   biClrUsed;//   位图实际使用的颜色表中的颜色数 
DWORD   biClrImportant;//   位图显示过程中重要的颜色数 
}   BITMAPINFOHEADER; 

4.   颜色表 

  颜色表用于说明位图中的颜色,它有若干个表项,每一个表项是一个RGBQUAD类型的结构,定义一种颜色。RGBQUAD结构的定义如下: 
typedef   struct   tagRGBQUAD   { 
BYTErgbBlue;//   蓝色的亮度(值范围为0-255) 
BYTErgbGreen;   //   绿色的亮度(值范围为0-255) 
BYTErgbRed;   //   红色的亮度(值范围为0-255) 
BYTErgbReserved;//   保留,必须为0 
}   RGBQUAD; 
颜色表中RGBQUAD结构数据的个数有biBitCount来确定: 
当biBitCount=1,4,8时,分别有2,16,256个表项; 
当biBitCount=24时,没有颜色表项。 
位图信息头和颜色表组成位图信息,BITMAPINFO结构定义如下: 
typedef   struct   tagBITMAPINFO   { 
BITMAPINFOHEADER   bmiHeader;   //   位图信息头 
RGBQUAD   bmiColors[1];   //   颜色表 
}   BITMAPINFO; 

5.   位图数据 
  位图数据记录了位图的每一个像素值,记录顺序是在扫描行内是从左到右,扫描行之间是从下到上。位图的一个像素值所占的字节数: 
当biBitCount=1时,8个像素占1个字节; 
当biBitCount=4时,2个像素占1个字节; 
当biBitCount=8时,1个像素占1个字节; 
当biBitCount=24时,1个像素占3个字节; 
Windows规定一个扫描行所占的字节数必须是 
4的倍数(即以long为单位),不足的以0填充, 
一个扫描行所占的字节数计算方法: 
DataSizePerLine=   (biWidth*   biBitCount+31)/8;   
//   一个扫描行所占的字节数 
DataSizePerLine=   DataSizePerLine/4*4;   //   字节数必须是4的倍数 
位图数据的大小(不压缩情况下): 
DataSize=   DataSizePerLine*   biHeight; 

时间: 2024-12-30 17:45:59

C语言编写的bmp读写程序的相关文章

c语言-用C语言编写一个对数组排序的程序,要求使用递归算法实现。

问题描述 用C语言编写一个对数组排序的程序,要求使用递归算法实现. 用C语言编写一个对数组排序的程序,要求使用递归算法实现. 解决方案 http://blog.sina.com.cn/s/blog_70441c8e0100pxuh.html 快速排序 解决方案二: http://blog.csdn.net/meditator_hkx/article/details/49756199 快速排序采用的二分法,正是将问题化大为小,展现递归的精髓-

Java与XML(二)用java编写xml的读写程序

xml|程序 Java与XML(二)用java编写xml的读写程序 这是读取xml文件的java程序,我调试好的.采用的是dom方式读取xml文件到Vector中.package src;import java.io.*;import java.util.Vector;import javax.xml.parsers.*;import org.w3c.dom.*;public class readxml { static Document document; private boolean va

用C语言编写自动排班程序

问题描述 用C语言编写自动排班程序 学校组织有需要值班的地方,而需要在学生没课的时候安排值班,这是一个繁琐的工程,我想写一个自动排班的程序.比如有50个人,每个人都有自己的课表,把自己没课的时间都按照一定要求列出来,然后按照一天上午两个班,下午两个班,晚上两个班,进行排班,一个班两个人,如果出现了有一个班所有同学都上课,那就空出来,其他的进行随机安排即可,一个人一周两个班.求大神指点!有偿1851841831 微信hao_zi7 解决方案 Linux下编写C语言程序 解决方案二: 不知道优先队列

输出-求用c语言编写这么一段程序

问题描述 求用c语言编写这么一段程序 1.用随机产生n个整数,赋给a数组 2.从小到大排序 3.计算最小数的阶乘 4.输出数组内容 解决方案 这个不难,就是先随机n个数放到数组再对数组排序,然后对数组第一个数求阶乘输出,最后输出整个排序过的数组,一步步来很简单 解决方案二: 参考代码段:https://github.com/707wk/Senior-middle-school/blob/master/Filling%20in%20the%20gaps.c 解决方案三: 你需要什么样的排序方式?最

用c语言编写的音频播放程序中的问题,注意这个是嵌入式程序

问题描述 用c语言编写的音频播放程序中的问题,注意这个是嵌入式程序 我得到一个用c语言编写的音频播放程序,但其中有两段代码看不懂,请各位大神帮帮忙,程序如下 #define mru_swap_16(x) ((unsigned short)( (((unsigned short)(x) & (unsigned short)0x00ffU) << 8) | (((unsigned short)(x) & (unsigned short)0xff00U) >> 8) ))

小弟想做一个 用C#语言编写的通讯录小程序。以SQL Server 2000做数据库。

问题描述 用SQLServer2000创建数据库的表,用C#语言基于Windowsr窗体做介面.输入名字,按"查找"按钮可以链接到数据库查找通讯录表的个人通讯.和输入学号和密码,按"查找"按钮可以链接到数据库成绩表的人个成绩. 解决方案 解决方案二:幫頂﹗﹗﹗幫頂﹗﹗﹗解决方案三:那就做吧.找人做?除非你给美金我做解决方案四:又不大,为啥不用access呢,单文件,方便.lz有什么问题吗?解决方案五:想做就做吧,难道有人阻止么?解决方案六:不知道楼主想表达什么意思?

用perl语言编写的sFlow解码程序:错误提示为:can&amp;amp;#x27;t bind:IO::Socket::INET: Unknown erro 如何解决该问题??

问题描述 #!/usr/bin/perluseNet::sFlow;useIO::Socket;my$sock=IO::Socket::INET->new(LocalPort=>'6343',Proto=>'udp')ordie"can'tbind:$@n";while($sock->recv($packet,1548)){&processPacket($packet);}die"Socketrecv:$!";subprocessPa

c语言-用C语言编写能进行四则运算的程序

问题描述 用C语言编写能进行四则运算的程序 我想用C语言编一个能进行简单的四则运算的程序,应该怎么写,请详细解释一下输入和计算部分 解决方案 实现四则运算是比较简单的,你可以使用两个栈,一个栈用来存贮操作的数字(0~n),一个栈用来存储操作符(+-等).你先把四则运算表达式存为一个字符串,然后遍历各个字符,如果是数字,那就存起来,如果是操作符,那就与前一个操作符比较,看看优先级是不是比前一个大,如果是,那就存起来,如果不是,那就把数字栈里的前两个数按照前一个操作符进行运算.结果用一个变量保存起来

中文字符-如何用C语言编写一个简单的输入法程序,要求可以输入汉字。

问题描述 如何用C语言编写一个简单的输入法程序,要求可以输入汉字. 不太清楚汉字在计算机中是如何存储的,想知道例如微软的智能ABC以及搜狗输入法是怎样实现拼音拼写下的汉字输入. 解决方案 首先要有一个汉字的编码库,比如GB2312编写的是拼音输入法的话,还要建立一个拼音与汉字对应的数据库然后根据用户输入的拼音,提示出对应的汉字(汉字的优先顺序由数据库决定,同时还可以学习该用户的使用习惯)如果输入法还支持智能联想输入的话,还要加入词库(也有优先级),这样可以根据前一个字来推断出下一个可能的字 解决