50分求单色BMP的缩放

圣迪 2009-02-16 02:48:30
因为项目原因,需要有对单色的BMP图像进行缩放,但是查了很多资料都是单色以上的色彩图像的缩放,所以很大一部分是用双线性插值算法的。
而单色在这里不适用,所以放分求高手给下代码,C/c++代码吧。我自己用最近领域法写的总是有出现像素不对的情况~~~~
...全文
232 14 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
14 条回复
切换为时间正序
请发表友善的回复…
发表回复
lebao82 2012-04-12
  • 打赏
  • 举报
回复
试了一下下面代码,缩小图片为137*83是没问题的,同时保存的还是位深度为1.

void SavePicture(CString strFileFrom, CString strFileTo)
{
CFileStatus status;
if(!CFile::GetStatus(strFileFrom,status))
{
AfxMessageBox("文件 " + strFileFrom + " 不存在!");
return;
}
CBitmap frombmp;
HBITMAP bmpHandle = (HBITMAP)LoadImage(NULL,strFileFrom,IMAGE_BITMAP,0,0,LR_LOADFROMFILE);
frombmp.Attach(bmpHandle);
//frombmp.Attach(ShrinkBitmap(bmpHandle,32,32));
//SaveBitmapToFile(frombmp,"logo1.bmp");
//CPaintDC dc(this); //绘制图象用

CBitmap BmpBack;
BmpBack.Attach(bmpHandle);//加载位图,IDB_XXXX
CDC dcMem;//载入大位图
dcMem.CreateCompatibleDC(NULL) ;
CBitmap* pOldBitmap = dcMem.SelectObject(&BmpBack);
CBitmap BmpTemp;//存储切割后的位图
BmpTemp.CreateCompatibleBitmap(&dcMem,137,83);
CDC DcTemp;
DcTemp.CreateCompatibleDC(NULL);
CBitmap* pOldBmpTemp = DcTemp.SelectObject(&BmpTemp);

BITMAP bm;
BmpBack.GetBitmap(&bm);

//下面代码可以实现缩放,
int iOldMode = SetStretchBltMode(DcTemp, COLORONCOLOR); //该语句可以解决失真问题

DcTemp.StretchBlt(0,0,137,83,&dcMem,0,0,bm.bmWidth,bm.bmHeight,SRCCOPY);

SetStretchBltMode(DcTemp, iOldMode);

//////////////////////////////////////
SaveBitmapToFile((HBITMAP)BmpTemp,strFileTo);

DcTemp.DeleteDC();//删除DC,否则Bitmap不可同时使用
dcMem.DeleteDC();//
}

bool SaveBitmapToFile(HBITMAP hBitmap, LPCTSTR lpFileName)
{
HDC hDC; //设备描述表
int iBits; //当前显示分辨率下每个像素所占字节数
WORD wBitCount; //位图中每个像素所占字节数
DWORD dwPaletteSize=0, //定义调色板大小, 位图中像素字节大小 ,位图文件大小 , 写入文件字节数
dwBmBitsSize,
dwDIBSize, dwWritten;
BITMAP Bitmap; //位图属性结构
BITMAPFILEHEADER bmfHdr; //位图文件头结构
BITMAPINFOHEADER bi; //位图信息头结构
LPBITMAPINFOHEADER lpbi; //指向位图信息头结构

HANDLE fh, hDib, hPal,hOldPal=NULL; //定义文件,分配内存句柄,调色板句柄

//计算位图文件每个像素所占字节数
HDC hWndDC = CreateDC("DISPLAY",NULL,NULL,NULL);
hDC = ::CreateCompatibleDC( hWndDC ) ;
int bitspixel=GetDeviceCaps(hDC, BITSPIXEL);
int planes= GetDeviceCaps(hDC, PLANES);
iBits = bitspixel* planes;
//iBits=1;
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 = 1 ;

//计算调色板大小
if (wBitCount <= 8)
dwPaletteSize = (1 << wBitCount) * 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 = wBitCount;
bi.biCompression = BI_RGB;
bi.biSizeImage = 0;
bi.biXPelsPerMeter = 0;
bi.biYPelsPerMeter = 0;
bi.biClrUsed = 0;
bi.biClrImportant = 0;

dwBmBitsSize = ((Bitmap.bmWidth * wBitCount+31)/32) * 4 * Bitmap.bmHeight ;

//为位图内容分配内存
hDib = GlobalAlloc(GHND,dwBmBitsSize+dwPaletteSize+sizeof(BITMAPINFOHEADER));
lpbi = (LPBITMAPINFOHEADER)GlobalLock(hDib);
*lpbi = bi;

// 处理调色板
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;

// 设置位图文件头
bmfHdr.bfType = 0x4D42; // "BM"
dwDIBSize = sizeof(BITMAPFILEHEADER)
+ sizeof(BITMAPINFOHEADER)
+ dwPaletteSize + dwBmBitsSize;
bmfHdr.bfSize = dwDIBSize;
bmfHdr.bfReserved1 = 0;
bmfHdr.bfReserved2 = 0;
bmfHdr.bfOffBits = (DWORD)sizeof(BITMAPFILEHEADER)
+ (DWORD)sizeof(BITMAPINFOHEADER)
+ dwPaletteSize;

// 写入位图文件头
WriteFile(fh, (LPSTR)&bmfHdr, sizeof(BITMAPFILEHEADER), &dwWritten, NULL);

// 写入位图文件其余内容
WriteFile(fh, (LPSTR)lpbi, dwDIBSize,
&dwWritten, NULL);

//清除
GlobalUnlock(hDib);
GlobalFree(hDib);
CloseHandle(fh);

return TRUE;
}
lebao82 2012-04-09
  • 打赏
  • 举报
回复
顶一下,恰好也在做这方面的东西,缩放倒是可以实现,现在的关键是缩放后图片不是单色的了
kacy16 2009-06-10
  • 打赏
  • 举报
回复
我现在也碰到这问题,是放大单色的bmp图,比较好的是 高度与宽度方向上都放大4倍,自己想了想,感觉好难啊,
因为针对每个像素点(也就是每个bit)都要处理,好象不比彩色的处理简单。不知道各位有啥好思路?
chiwa737 2009-05-26
  • 打赏
  • 举报
回复
Imagemagick 有源代码, 看看就好了
要么你就直接用imagemagick好了, 里面有magic++ 专门给 c++程序员用得
libin009 2009-05-26
  • 打赏
  • 举报
回复
楼上都是高手,学些!。。。。。
diannaomingong 2009-05-15
  • 打赏
  • 举报
回复
if((BitmapTemp->GetWidth()!=w)|| (BitmapTemp->GetHeight()!=h))
{
wxImage *drawImg = new wxImage(BitmapTemp->ConvertToImage());
drawImg->Rescale(w,h);
if ( BitmapTemp )
{
delete BitmapTemp;
BitmapTemp = new wxBitmap(drawImg->Copy(),-1);
}
delete drawImg;
drawImg = (wxImage *)NULL;
}


zhoujk 2009-05-14
  • 打赏
  • 举报
回复
转到 8位,缩放以后,再转回1BIT
laoma_hbu 2009-04-06
  • 打赏
  • 举报
回复
单色位图也可以用双线性插值,只是取像素值有点麻烦。
BOOL ImageScale(CImage &imgSrc, CImage &imgDst, double alpha)
{
if (alpha < 1)
{
return FALSE;
}
int nHeight = imgSrc.GetHeight();
int nWidth = imgSrc.GetWidth();
int nLineBytes = imgSrc.GetLineSizeInBytes();
LPBYTE lpDIBBits = imgSrc.LockRawImg();

int nNewWidth = nWidth / alpha;
int nNewHeight = nHeight / alpha;
int nNewLineBytes = (nNewWidth + 31) / 32 * 4;

imgDst.Initialize(nNewWidth, nNewHeight, 1, 1);

LPBYTE lpNewDIBBits = imgDst.LockRawImg();
memset(lpNewDIBBits, 0, nNewLineBytes * nNewHeight);

int i, j;
LPBYTE lpSrc, lpDst;

for (i = 0; i < nNewHeight; i++)
{
for (j =0; j < nNewWidth; j++)
{
double q = i * alpha;
double p = j * alpha;

int y1, y2;
y1 = (int)q;
y2 = y1 + 1;
q = q - y1;

int x1, x2;
x1 = (int)p;
x2 = x1+1;
p = p - x1;

if ((x2 >nWidth) || (y2 >nHeight))
continue;
if (x2 == nWidth)
x2 = x2 -1 ;
if (y2 == nHeight)
y2 = y2 -1;

double a = 0.0, b = 0.0, c = 0.0, d = 0.0;

a = (double)GetPixValue(lpDIBBits, nLineBytes, nHeight, y1, x1);
b = (double)GetPixValue(lpDIBBits, nLineBytes, nHeight, y1, x2);
c = (double)GetPixValue(lpDIBBits, nLineBytes, nHeight, y2, x1);
d = (double)GetPixValue(lpDIBBits, nLineBytes, nHeight, y2, x2);

double t1,t2,t3;
t1 = (1-p)*a + p*b;
t2 = (1-p)*c + p*d;

t3 = (1-q)*t1 + q*t2;

if (t3>0.5) {
SetPixValue(lpNewDIBBits, nNewLineBytes, nNewHeight, i, j, 1);
}
else{
SetPixValue(lpNewDIBBits, nNewLineBytes, nNewHeight, i, j, 0);
}
}
}

imgSrc.UnlockRawImg();
imgDst.UnlockRawImg();
return TRUE;
}

BYTE GetPixValue(LPBYTE lpDIBBits, int nLineBytes, int nHeight, int i, int j)
{
LPBYTE lpSrc = lpDIBBits + nLineBytes * (nHeight - 1 - i) + j/8;
BYTE pix = (BYTE)*lpSrc;
BYTE pos = (BYTE)(7 - j%8);
pix &= (0x01<<pos);
return (BYTE)(pix >> pos);
}
void SetPixValue(LPBYTE lpDIBBits, int nLineBytes, int nHeight, int i, int j, BYTE pixel)
{
LPBYTE lpDst = lpDIBBits + nLineBytes * (nHeight - 1 - i) + j/8;
BYTE pix = (BYTE)*lpDst;
BYTE pos = (BYTE)(7 - j%8);
*lpDst &= ~(0x01<<pos);
*lpDst |= ((pixel & 0x01)<<pos);
return;
}
laviewpbt 2009-04-05
  • 打赏
  • 举报
回复
楼上的这些人都是理论家,单色和索引色的缩放如果你直接写代码的话要比彩色的难很多,你们知道扫描行对齐吗,知道索引色的数据代表的意义吗?正真意义上的单色缩放是很困难的,因为单色图像在内存中一个像素是用一个位表示的,不是一个字节。
bafghgle 2009-04-02
  • 打赏
  • 举报
回复
这个是不难的。。
crond123 2009-03-20
  • 打赏
  • 举报
回复
不难,直接用OpenCV就可以简单搞定
直接写的话单色比彩色还要简单
civanx 2009-03-12
  • 打赏
  • 举报
回复
不难

http://search.codesoso.com/Search?q=BMP的缩放
实达诚实 2009-03-06
  • 打赏
  • 举报
回复
很简单的,我上传了一个缩放代码,你去下载吧
youxia000 2009-02-16
  • 打赏
  • 举报
回复
不应该啊,缩放算法 双线性也能用啊, 不过放大效果比较好的 双三次 bicubic 方法,你搜下,不算难写

另外 你的这个单色是不是用调色板配色的?那样的你插值恐怕是不对,这种情况建议你转到RGB空间做

灰度 RGB的 用双线性都没问题 就是说效果好坏 是否满足你要求了

4,498

社区成员

发帖
与我相关
我的任务
社区描述
图形图像/机器视觉
社区管理员
  • 机器视觉
  • 迪菲赫尔曼
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧