24位真彩色转换为8位灰度图片(完整代码)

#include <windows.h>
#include<iostream>

using namespace std;
BOOL BMP24to8(char *szSourceFile,char *szTargetFile);

int main(int argc,char* argv[])
{
    //调用这个函数直接把24位真彩色灰度化
    BOOL stat=BMP24to8("D://filein.bmp","D://fileout.bmp");
    return 0;
}
BOOL BMP24to8(char *szSourceFile,char *szTargetFile)
{
    HANDLE hSourceFile=INVALID_HANDLE_VALUE,hTargetFile=INVALID_HANDLE_VALUE;
    DWORD dwSourceSize=0,dwTargetSize=0;
    PBYTE pSource=NULL,pTarget=NULL;
    hSourceFile=CreateFile(szSourceFile,GENERIC_READ,FILE_SHARE_READ,NULL,
    OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL,NULL);
    if(hSourceFile==INVALID_HANDLE_VALUE)
    {
        cout<<"打开文件失败!\n";
        return FALSE;
    }
    dwSourceSize=GetFileSize(hSourceFile,NULL);
    pSource=(PBYTE)VirtualAlloc(NULL,dwSourceSize,MEM_COMMIT,PAGE_READWRITE);
    //分配空间失败或者文件太小(BMP文件不可能小于54个字节)

    if(pSource==NULL||dwSourceSize<=54)
    {
        CloseHandle(hSourceFile);
        cout<<"分配空间失败!\n";
        return FALSE;
    }
    DWORD dwTemp=0;
    ReadFile(hSourceFile,pSource,dwSourceSize,&dwTemp,NULL);
    BITMAPFILEHEADER *pSourceFileHeader=(BITMAPFILEHEADER*)pSource;
    BITMAPINFOHEADER *pSourceInfoHeader=(BITMAPINFOHEADER*)(pSource+sizeof(BITMAPFILEHEADER));
    //不是BMP文件或者不是24位真彩色

    if(pSourceFileHeader->bfType!=0x4d42||pSourceInfoHeader->biBitCount!=24)
    {
        CloseHandle(hSourceFile);
        VirtualFree(pSource,NULL,MEM_RELEASE);
        cout<<"不是BMP文件!\n";
        return FALSE;
    }
    CloseHandle(hSourceFile);
    LONG nWidth=pSourceInfoHeader->biWidth;
    LONG nHeight=pSourceInfoHeader->biHeight;

    /*
    * 当biBitCount=24时,1个像素占3个字节。
    * BMP文件格式的对齐要求每行的数据的长度必须是4字节的倍数,如果不够需要进行比特填充(以0填充),这样可以达到按行的快速存取。
    */

    LONG nSourceWidth=nWidth*3;if(nSourceWidth%4) nSourceWidth=(nSourceWidth/4+1)*4;
    LONG nTargetWidth=nWidth;if(nTargetWidth%4) nTargetWidth=(nTargetWidth/4+1)*4;

    /*
    * biSizeImage:指定实际的位图数据所占 用的字节数,通过公式计算:
    * biSizeImage = biWidth’字节 * biHeight 字节,公式中的biWidth’字节必须 是4的整数倍
    */

    dwTargetSize=sizeof(BITMAPFILEHEADER)+sizeof(BITMAPINFOHEADER)+sizeof(RGBQUAD)*256+nHeight*nTargetWidth;
    pTarget=(PBYTE)VirtualAlloc(NULL,dwTargetSize,MEM_COMMIT,PAGE_READWRITE);
    memset(pTarget,0,dwTargetSize);
    if(pTarget==NULL)
    {
        VirtualFree(pTarget,NULL,MEM_RELEASE);
        cout<<"新建文件失败!\n";
        return FALSE;
    }
    BITMAPFILEHEADER *pTargetFileHeader=(BITMAPFILEHEADER *)pTarget;
    BITMAPINFOHEADER *pTargetInfoHeader =
    (BITMAPINFOHEADER *)(pTarget+sizeof(BITMAPFILEHEADER));
    pTargetFileHeader->bfType=pSourceFileHeader->bfType;
    pTargetFileHeader->bfSize=dwTargetSize;
    pTargetFileHeader->bfReserved1=0;
    pTargetFileHeader->bfReserved2=0;
    pTargetFileHeader->bfOffBits=sizeof(BITMAPFILEHEADER)+sizeof(BITMAPINFOHEADER)+sizeof(RGBQUAD)*256;
    pTargetInfoHeader->biBitCount=8;
    pTargetInfoHeader->biClrImportant=0;
    pTargetInfoHeader->biClrUsed=256;
    pTargetInfoHeader->biCompression=BI_RGB;
    pTargetInfoHeader->biHeight=pSourceInfoHeader->biHeight;
    pTargetInfoHeader->biPlanes=1;
    pTargetInfoHeader->biSize=sizeof(BITMAPINFOHEADER);
    pTargetInfoHeader->biSizeImage=nHeight*nTargetWidth;
    pTargetInfoHeader->biWidth=pSourceInfoHeader->biWidth;
    pTargetInfoHeader->biXPelsPerMeter=pSourceInfoHeader->biXPelsPerMeter;
    pTargetInfoHeader->biYPelsPerMeter=pSourceInfoHeader->biYPelsPerMeter;
    RGBQUAD *pRgb;
    for(int i=0;i<256;i++)//初始化8位灰度图的调色板信息
    {
        pRgb = (RGBQUAD*)(pTarget+sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER)+i*sizeof(RGBQUAD));
        pRgb->rgbBlue=i;pRgb->rgbGreen=i;pRgb->rgbRed=i;pRgb->rgbReserved=0;
    }
    for (int m=0;m<nHeight;m++)//转化真彩色图为灰度图
    {
        for(int n=0;n<nWidth;n++)
        {
                //将左下角区域编码
                if ((m > 30 && m < 40) && (n >30 && n <90)) continue;
                if ((m > 70 && m < 80) && (n > 30 && n < 90)) continue;
                if ((m > 110 && m < 120) && (n > 30 && n < 90)) continue;
                if ((m > 70 && m < 120) && (n >= 80 && n <= 90)) continue;
                if ((m > 30 && m < 80) && (n >= 30 && n <= 40)) continue;

                pTarget[pTargetFileHeader->bfOffBits+m*nTargetWidth+n] =
                pSource[pSourceFileHeader->bfOffBits+m*nSourceWidth+n*3]*0.114
                +pSource[pSourceFileHeader->bfOffBits+m*nSourceWidth+n*3+1]*0.587
                +pSource[pSourceFileHeader->bfOffBits+m*nSourceWidth+n*3+2]*0.299;
        }
    }
    hTargetFile = CreateFile(szTargetFile,GENERIC_WRITE,FILE_SHARE_WRITE,
    NULL,CREATE_ALWAYS,FILE_ATTRIBUTE_NORMAL,NULL);
    BOOL stat=WriteFile(hTargetFile,pTarget,dwTargetSize,&dwTemp,NULL);
    CloseHandle(hTargetFile);
    VirtualFree(pSource,NULL,MEM_RELEASE);
    VirtualFree(pTarget,NULL,MEM_RELEASE);
    return stat;
}

http://blog.sina.com.cn/s/blog_4c6e822d0102dyk8.html

附上优秀的参考文献链接。

  • 1
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值