当前位置:  技术问答>linux和unix

linux下24bitsBMP转灰度图,怎么处理,有代码,但是不知道哪里有误.

    来源: 互联网  发布时间:2016-06-21

    本文导语:  #include #include #pragma pack(2); typedef   struct   {   unsigned short     bfType;                         /*   文件类型,   必须为   "BM"(0x4D42) */   unsigned long   bfSize;                      ...

#include
#include
#pragma pack(2);

typedef   struct  
{  
unsigned short     bfType;                         /*   文件类型,   必须为   "BM"(0x4D42) */  
unsigned long   bfSize;                         /*   文件的大小(字节)   */  
unsigned short     bfReserved1;               /*   保留,   必须为   0   */  
unsigned short     bfReserved2;               /*   保留,   必须为   0   */  
unsigned long   bfoffBits;                   /*   位图阵列相对于文件头的偏移量(字节)   */  
}  BITMAPFILEHEADER;               /*   文件头结构   */  

typedef   struct  
{  
unsigned long   biSize;                         /*   size   of   BITMAPINFOHEADER   */  
unsigned long   biWidth;                       /*   位图宽度(像素)   */  
unsigned long   biHeight;                     /*   位图高度(像素)   */  
unsigned short  biPlanes;                     /*   目标设备的位平面数,   必须置为1   */  
unsigned short  biBitCount;                 /*   每个像素的位数,   1,4,8或24   */  
unsigned long   biCompress;                 /*   位图阵列的压缩方法,0=不压缩   */  
unsigned long   biSizeImage;               /*   图像大小(字节)   */  
unsigned long   biXPelsPerMeter;       /*   目标设备水平每米像素个数   */  
unsigned long   biYPelsPerMeter;       /*   目标设备垂直每米像素个数   */  
unsigned long   biClrUsed;                   /*   位图实际使用的颜色表的颜色数   */  
unsigned long   biClrImportant;         /*   重要颜色索引的个数   */  
}   BITMAPINFOHEADER;               /*   位图信息头结构   */  
typedef   struct   
{   
unsigned char  rgbBlue;   
unsigned char   rgbGreen;   
unsigned char   rgbRed;   
unsigned char rgbReserved;   
}   RGBQUAD;

typedef struct  
{
BITMAPINFOHEADER bmiHeader;
RGBQUAD bmiColors[256];
}BITMAPINFO, * PBITMAPINFO;

BITMAPFILEHEADER bfile, bbfile;
BITMAPINFOHEADER binfo, bbinfo;

BITMAPINFO bmi; //»Ò¶È͌ʱÓÃÕâžö.
PBITMAPINFO pbmi;
FILE *in, *out;

int open()
{
if((in = fopen("girl.bmp", "rb")) == NULL)
{
printf("Open file failed.n");
exit(-1);
}
if((out = fopen("gray_girl.bmp", "wb")) == NULL)
{
printf("Create copy file failed.n");
exit(-1);
}
return 0;
}

int init_bmpStruct()
{
fread(&bfile.bfType,   sizeof(bfile.bfType),   1,   in);  
fwrite(&bfile.bfType,   sizeof(bfile.bfType),   1,   out); 

fread(&bfile.bfSize,   sizeof(bfile.bfSize),   1,   in);
fwrite(&bfile.bfSize,   sizeof(bfile.bfSize),   1,   out);

fread(&bfile.bfReserved1,   sizeof(bbfile.bfReserved1),   1,   in);
fwrite(&bfile.bfReserved1,   sizeof(bfile.bfReserved1),   1,   out);

fread(&bfile.bfReserved2,   sizeof(bfile.bfReserved2),   1,   in);
fwrite(&bfile.bfReserved2,   sizeof(bfile.bfReserved2),   1,   out);

fread(&bfile.bfoffBits,   sizeof(bfile.bfoffBits),   1,   in);
fwrite(&bfile.bfoffBits,   sizeof(bfile.bfoffBits),   1,   out);

fread(&binfo.biSize,   sizeof(binfo.biSize),   1,   in); 
fwrite(&binfo.biSize,   sizeof(binfo.biSize),   1,   out); 

fread(&binfo. biWidth,   sizeof(binfo. biWidth),   1,   in); 
fwrite(&binfo.biWidth,   sizeof(binfo.biWidth),   1,   out);

fread(&binfo.biHeight,   sizeof(binfo.biHeight),   1,   in); 
fwrite(&binfo.biHeight,   sizeof(binfo.biHeight),   1,   out);

fread(&binfo.biPlanes,   sizeof(binfo.biPlanes),   1,   in); 
fwrite(&binfo.biPlanes,   sizeof(binfo.biPlanes),   1,   out);

fread(&binfo.biBitCount,   sizeof(binfo.biBitCount),   1,   in);
fwrite(&binfo.biBitCount,   sizeof(binfo.biBitCount),   1,   out);
 
fread(&binfo.biCompress,   sizeof(binfo.biCompress),   1,   in); 
fwrite(&binfo.biCompress,   sizeof(binfo.biCompress),   1,   out);

fread(&binfo.biSizeImage,   sizeof(binfo.biSizeImage),   1,   in); 
fwrite(&binfo.biSizeImage,   sizeof(binfo.biSizeImage),   1,   out);

fread(&binfo.biXPelsPerMeter,   sizeof(binfo.biXPelsPerMeter),   1,   in); 
fwrite(&binfo.biXPelsPerMeter,   sizeof(binfo.biXPelsPerMeter),   1,   out);

fread(&binfo.biYPelsPerMeter,   sizeof(binfo.biYPelsPerMeter),   1,   in); 
fwrite(&binfo.biYPelsPerMeter,   sizeof(binfo.biYPelsPerMeter),   1,   out);

fread(&binfo.biClrUsed,   sizeof(binfo.biClrUsed),   1,   in); 
fwrite(&binfo.biClrUsed,   sizeof(binfo.biClrUsed),   1,   out);

fread(&binfo.biClrImportant,   sizeof(binfo.biClrImportant),   1,   in);
fwrite(&binfo.biClrImportant,   sizeof(binfo.biClrImportant),   1,   out);

printf("biSize = %ldn",binfo.biSize); 
printf("biWidth = %ldn",binfo.biWidth);
printf("biHeight = %ldn",binfo.biHeight);
printf("biPlanes = %dn",binfo.biPlanes);
printf("biBitCount = %dn",binfo.biBitCount);
printf("biCompress = %dn",binfo.biCompress);
printf("biSizeImage = %ldn",binfo.biSizeImage);
printf("biXpersPerMeter = %ldn",binfo.biXPelsPerMeter);
printf("biYPersPerMeter = %ldn",binfo.biYPelsPerMeter);
printf("biClrUsed = %ldn",binfo.biClrUsed);
printf("biClrImportant = %ldn", binfo.biClrImportant);
return 0;

}


int WritePixtoFile()
{
int i = 0, j = 0;
        unsigned long bboffest = bbfile.bfoffBits;
unsigned long boffest = bfile.bfoffBits;
int nBufferWide = 4 * (( binfo.biWidth * 24 + 31 )/ 32); 
int bnBufferWide = 4 * (( binfo.biWidth * 24 + 31 )/ 32);
int Y;
unsigned char YY;
int R, G, B;
int RR, GG, BB;
long count = 0;
unsigned char *RGBbuffer = (unsigned char*)malloc(nBufferWide * binfo.biHeight);
unsigned char *bRGBbuffer = (unsigned char*)malloc(bnBufferWide * binfo.biHeight);


fseek(in, bboffest, SEEK_SET); ???这句定位是不是有问题?因为最后保存的图像一片黑.
fread(RGBbuffer, sizeof(unsigned char), nBufferWide * binfo.biHeight, in);

for(i = binfo.biWidth -1 ; i >= 0; i--)
{
for(j = binfo.biHeight - 1; j >=0 ; j--)
{

R = (int)(*(RGBbuffer + i*3 + j));
G = (int)(*(RGBbuffer + i*3 + j + 1));
B = (int)(*(RGBbuffer+ i*3 + j + 2));

Y = ((66 * R + 129 * G + 25 * B + 128)>>8) + 16;
YY = (unsigned char) ((Y255)?255:Y));
RR = YY;
GG = YY;
BB = YY;
printf("YY = %d,", YY);
*(bRGBbuffer + i* 3 +j) = RR;
*(bRGBbuffer + i* 3 +j +1) = RR;
*(bRGBbuffer + i* 3 +j +2) = RR;

//memcpy(bRGBbuffer++, &YY, sizeof(unsigned char));

//printf("YY = %u, count = %ldn", YY, count);
}
}
fwrite(bRGBbuffer ,1 , bnBufferWide * binfo.biHeight,out);
}

int main()
{
open();
init_bmpStruct();

WritePixtoFile();
fclose(in);
fclose(out);
printf("BITMAPFILEHEADER size = %d,BITMAPINFOHEADER size = %d, RGBQUAD size = %dn ", sizeof(BITMAPFILEHEADER), sizeof(BITMAPINFOHEADER), sizeof(RGBQUAD));
return 1;
}

girl.bmp是一个24位的BMP图像,320*240的.
但是保存的gray_girl.bmp,就是一片黑啊~

是不是offset定位错了?但是不知道怎么错的.
哪位朋友知道啊.我是新手.点指一下.非常感激!

|
不是吧 写得这么复杂?
我写的 uClinux 下 Frame Buffer 驱动显示 24 位图片的程序

typedef struct    /* 24 bit BMP color */ 

unsigned char b; 
unsigned char g; 
unsigned char r; 
} ColorBGR24BIT;

/*
 * 不处理图片是否压缩
 * 从下面往上面开始显示
 * ScreenWidth、ScreenHeight 分别为屏幕大小和高度
*/
void disp_24bit_bmp(char *filename)
{
int width, height;
short int bitcount;
FILE *fp;
ColorBGR24BIT *colorRead24; /* read the color data(24 bit) */
unsigned int tempcolor;
int i, j;
unsigned int *pD;
unsigned char r, g, b;

if((fp=fopen(filename, "rb"))==NULL) 
{
printf("Read %s error.n", filename);
return;
}

fseek(fp, 18, SEEK_SET);
fread(&width, sizeof(width), 1, fp);
fread(&height, sizeof(height), 1, fp);
fseek(fp, 28, SEEK_SET);
fread(&bitcount, sizeof(bitcount), 1, fp);

if(24 !=bitcount)
{
fclose(fp);
printf("Only support 24 bitmap file.n");
return;
}

/* color * width memory */
colorRead24 = (ColorBGR24BIT *)malloc(width*sizeof(ColorBGR24BIT)); 

fseek(fp, 54, SEEK_SET);     /* color data */

for(j=height-1; j>=0; j--)
{
/* fread(colorRead24, width*sizeof(ColorBGR24BIT), 1, fp); */
fread(colorRead24, width*3, 1, fp);

if(width%4) fseek(fp, (width%4), SEEK_CUR); /* fix width and file pointer 24-bits */

if(j >= ScreenHeight) continue;

for(i=0; i= ScreenWidth) break;
r = colorRead24[i].r;
g = colorRead24[i].g;
b = colorRead24[i].b;
tempcolor = RGB(r, g, b);
pD = (unsigned int*)framebase + (j * ScreenWidth+ i); /* 指向 Frame Buffer 内核空间 */
*pD = tempcolor; /* 赋值(32位色) */
}
}
free(colorRead24);

fclose(fp);

printf("Read %s ok, Width=%d, Height=%d.n", filename, width, height);
}

|
Y = ((66 * R + 129 * G + 25 * B + 128)>>8) + 16;  
是错误的 66+129+25=220 不是 256(2^8)
灰度 = 红×0.299 + 绿×0.587 + 蓝×0.114

为了不计算浮点数(快速)
Y = ((76 * R + 150 * G + 29 * B + 128)>>8) 

|
没有环境,没法调
是不是字节对齐的问题

|
R = (int)(*(RGBbuffer + i*3 + j)); 
G = (int)(*(RGBbuffer + i*3 + j + 1)); 
B = (int)(*(RGBbuffer+ i*3 + j + 2)); 

Y = ((66 * R + 129 * G + 25 * B + 128)>>8) + 16; 

R,G,B这里应该用char。
Y啊,这个看不懂。

    
 
 
 
本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
本站(WWW.)站内文章除注明原创外,均为转载、整理或搜集自网络。欢迎任何形式的转载,转载请注明出处。












  • 相关文章推荐
  • Linux c++虚函数(virtual function)简单用法示例代码
  • 读懂 Linux 内核代码不难,难的是读懂 Linux 内核代码背后的哲学!
  • Linux c++库boost unordered_set数据插入及查找代码举例
  • 为什么比较freebsd和linux的源代码,觉得linux的代码十分的庞大。。
  • Linux c++库boost unordered_map数据插入及查找代码举例
  • linux源代码的时间函数源代码
  • Linux下c函数dlopen实现加载动态库so文件代码举例
  • RedHat Linux是开放源代码的吗?如何能看见它的源代码,如何进行修改?
  • linux c 生成随机数srand函数和rand函数介绍及代码示例
  • windows下写的代码 gb2312 如何转成 LINUX和WINDOWS都可正常显示的代码
  • Linux 共享内存介绍及实现代码
  • Linux源代码中ata驱动为什么要调用pci驱动的代码
  • SSL握手通信详解及linux下c/c++ SSL Socket代码举例
  • 高分请教!Netscape浏览器的源代码公开了吗?从那里可以得到For linux的源代码?
  • 谁能告诉我早期LINUX内核的源代码和C编译器的源代码,哪能找到!谢谢!要多少分都给!
  • 谁有最早版本的LINUX源代码?能发给我一份吗?高分求救啊……我真的很需要最早的一些版本的源代码的!
  • 高深问题:有了linux内核源代码如何做成一个linux操作系统
  • 哪本linux源码分析的书里详细讲解了Linux的核心启动代码
  • 我们知道内核源代码文件都放在/linux下,但目录/linux的绝对路径是啥?谢谢!
  • 我现在正读Linux源代码,请问下面代码是什么意思?
  • 想使用Kliyx把Delphi写的代码编译为Linux程序, 装什么Linux什么版本最好?请明人指教,谢谢!?
  • linux c/c++ IP字符串转换成可比较大小的数字
  • 在win分区上安装linux和独立分区安装linux有什么区别?可以同时安装吗?(两个linux系统)
  • linux哪个版本好?linux操作系统版本详细介绍及选择方案推荐
  • 在虚拟机上安装的linux上,能像真的linux系统一样开发linux程序么?
  • secureCRT下Linux终端汉字乱码解决方法
  • 我重装window后,把linux的引导区覆盖了,进不了linux怎么办?急啊,望热心的人帮助 (现在有linux的盘)
  • Linux c字符串中不可打印字符转换成16进制
  • 安装vmware软件,不用再安装linux系统,就可以模拟linux系统了,然后可以在其上学习一下LINUX下的基本操作 了?
  • Linux常用命令介绍:更改所属用户群组或档案属性
  • 红旗Linux主机可以通过127.0.0.1访问,但如何是连网的Win2000机器通过Linux的IP去访问Linux


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3