我在将RGB彩色图转变成深度为8bit的灰度图时,结果显示出来却是全是黑的?不知道我的程序那里有问题?
我在将RGB彩色图转变成深度为8bit的灰度图时,结果显示出来却是全是黑的?不知道我的程序那里有问题?下面是我的代码,
int CDispayDlg::SaveBitmapToFile(HBITMAP hBitmap, LPSTR lpFileName)
{
RGBQUAD rgb[256];
//lpFileName 为位图文件名
HDC hDC;
//设备描述表
int i,j, iBits;
//当前显示分辨率下每个像素所占字节数
WORD wBitCount;
//位图中每个像素所占字节数
//定义调色板大小, 位图中像素字节大小 , 位图文件大小 , 写入文件字节数
DWORD dwPaletteSize=0,dwPaletteSize1=0, dwBmBitsSize,dwBmBitsSize1, dwDIBSize, dwWritten;
BITMAP Bitmap;
//位图属性结构
BITMAPFILEHEADER bmfHdr;
//位图文件头结构
BITMAPINFOHEADER bi,ai;
//位图信息头结构
LPBITMAPINFOHEADER lpbi,lpai;
// LPBITMAPINFOHEADER k;
//指向位图信息头结构
HANDLE fh, hDib,hDib1, hPal,hOldPal=NULL;
//定义文件,分配内存句柄,调色板句柄
//计算位图文件每个像素所占字节数
hDC = CreateDC("DISPLAY",NULL,NULL,NULL);
iBits = 16 ;//GetDeviceCaps(hDC, BITSPIXEL) * GetDeviceCaps(hDC, PLANES);
DeleteDC(hDC);
if (iBits <= 1)
wBitCount = 1;
else if (iBits <= 4)
wBitCount = 4;
else if (iBits <= 8)
wBitCount = 8;
else if (iBits <= 24)
wBitCount = 24;
else
wBitCount = 32;
//计算调色板大小
if (wBitCount <= 8)
dwPaletteSize = (1 << wBitCount) * sizeof(RGBQUAD);
dwPaletteSize1 = (1 << 8) * sizeof(RGBQUAD);
//设置位图信息头结构
GetObject(hBitmap, sizeof(BITMAP), (LPSTR)&Bitmap);
bi.biSize = sizeof(BITMAPINFOHEADER);
bi.biWidth = Bitmap.bmWidth;
bi.biHeight = Bitmap.bmHeight;
bi.biPlanes = 1;
bi.biBitCount = 24;
bi.biCompression = BI_RGB;
bi.biSizeImage = 0;
bi.biXPelsPerMeter = 0;
bi.biYPelsPerMeter = 0;
bi.biClrUsed = 0;
bi.biClrImportant = 0;
/////////////////////////////////////////////
ai.biSize = sizeof(BITMAPINFOHEADER);
ai.biWidth = Bitmap.bmWidth;
ai.biHeight = Bitmap.bmHeight;
ai.biPlanes = 1;
ai.biBitCount = 8;
ai.biCompression = BI_RGB;
ai.biSizeImage = Bitmap.bmWidth*Bitmap.bmHeight;
ai.biXPelsPerMeter = 0;
ai.biYPelsPerMeter = 0;
ai.biClrUsed = 0;
ai.biClrImportant = 0;
////////////////////////////////////////////////////
for(i=0;i<256;i++)
{
rgb[i].rgbBlue=i;
rgb[i].rgbGreen=i;
rgb[i].rgbRed=i;
rgb[i].rgbReserved=0;
}
int height;
int width;
width=bi.biWidth;
height=bi.biHeight;
// scanwidth=((Bitmap.bmWidth * wBitCount+31)/32);
dwBmBitsSize = ((Bitmap.bmWidth * wBitCount+31)/32)* 4 *Bitmap.bmHeight ;
dwBmBitsSize1 = Bitmap.bmWidth *Bitmap.bmHeight ;
//为位图内容分配内存
hDib = GlobalAlloc(GHND,dwBmBitsSize+dwPaletteSize+sizeof(BITMAPINFOHEADER));
hDib1 = GlobalAlloc(GHND,dwBmBitsSize1+dwPaletteSize1+sizeof(BITMAPINFOHEADER));
lpbi = (LPBITMAPINFOHEADER)GlobalLock(hDib);
lpai = (LPBITMAPINFOHEADER)GlobalLock(hDib1);
*lpbi = bi;
*lpai = ai;
// 处理调色板
hPal = GetStockObject(DEFAULT_PALETTE);
if (hPal)
{
hDC = ::GetDC(NULL);
hOldPal = SelectPalette(hDC, (HPALETTE)hPal, FALSE);
RealizePalette(hDC);
}
// 获取该调色板下新的像素值
GetDIBits(hDC, hBitmap, 0, (UINT) Bitmap.bmHeight,
(LPSTR)lpbi + sizeof(BITMAPINFOHEADER)+dwPaletteSize,
(LPBITMAPINFO)lpbi, DIB_RGB_COLORS);
// 恢复调色板
if (hOldPal)
{
SelectPalette(hDC, (HPALETTE)hOldPal, TRUE);
RealizePalette(hDC);
::ReleaseDC(NULL, hDC);
}
// 创建位图文件
fh = CreateFile(lpFileName, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS,
FILE_ATTRIBUTE_NORMAL | FILE_FLAG_SEQUENTIAL_SCAN, NULL);
if (fh == INVALID_HANDLE_VALUE)
return FALSE;
// 设置位图文件头
dwDIBSize = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + dwPaletteSize1 +dwBmBitsSize1;
bmfHdr.bfType = 0x4D42; // "BM"
bmfHdr.bfSize = dwDIBSize;
bmfHdr.bfReserved1 = 0;
bmfHdr.bfReserved2 = 0;
bmfHdr.bfOffBits = (DWORD)sizeof(BITMAPFILEHEADER)+ (DWORD)sizeof(BITMAPINFOHEADER)+ dwPaletteSize1;
// 写入位图文件头
WriteFile(fh, (LPSTR)&bmfHdr, sizeof(BITMAPFILEHEADER), &dwWritten, NULL);
for(j=0;j<256*256*3-3;j+=3)
{
BYTE bBlue =(BYTE) *((LPSTR)lpbi + sizeof(BITMAPINFOHEADER)+dwPaletteSize+j + 0);
BYTE bGreen =(BYTE) *((LPSTR)lpbi + sizeof(BITMAPINFOHEADER)+dwPaletteSize+j + 1);
BYTE bRed =(BYTE) *((LPSTR)lpbi + sizeof(BITMAPINFOHEADER)+dwPaletteSize+j + 2);
BYTE Temp;
// Temp=(BYTE)(bBlue*0.144+bGreen*0.588+bRed*0.299+0.5);
Temp=(BYTE) ((117 * bBlue + 601 * bGreen + 306* bRed+ 512 ) >> 10 ); //修改
if(Temp>255)
Temp=255;
*((LPSTR)lpai + sizeof(BITMAPINFOHEADER)+dwPaletteSize1+j/3)=Temp;
}
// 写入位图文件其余内容
WriteFile(fh, (LPSTR)lpai, dwDIBSize, &dwWritten, NULL);
/****************************************************************/
//清除
GlobalUnlock(hDib);
GlobalFree(hDib);
GlobalUnlock(hDib1);
GlobalFree(hDib1);
CloseHandle(fh);
return 0;
}
lpai用来开获得的8比特单色图,我试过用fwrite将上面的像素值写入文件,可以成功写入,得到一个转换后的8比特灰度图,但是当我改用WriteFile函数时,结果显示的是一幅8比特的全黑图像,不知是什么原因,那位大虾能否帮我分析分析。还有一个问题,要是我想将24位的RGB转换成8比特的位图时,但结果仍然是彩色的,怎么做?