bmp 缩放代码(BMP位图任意放大 和缩小)
前几天碰上需要对bmp位图进行缩放的功能,
调用API函数,虽然能实现位图缩放,但是对有放大的效果好,缩小会造成失真,图像上有花点,让人难以接受
,因为本人以前学易语言,易语言有一段代码,对bmp图像缩放效果非常好,
昨天抽空,把它翻译成c++代码了,经验证,非常好用,帖上代码:
这一段是对宽度进行缩放:
void__stdcallbmpsetH(unsignedchar*bitmap,inth,unsignedchar*bitmap2)
{
inti,j,k,bitmapH1,bitmapH2,bitmapV1,bitmapV2,bitmapSize1,bitmapSize2,bitmapLen1,bitmapLen2;
intstartH1,startH2,R,G,B,R1,G1,B1;
// 位图宽1=取字节集数据(位图数据,#整数型,19)
bitmapH1=*((int*)(bitmap+18));
bitmapV1=*((int*)(bitmap+22));
intredress1=bitmapH1%4;
intbitmapNH1=bitmapH1*3+redress1;
bitmapLen1=bitmapNH1*bitmapV1;
bitmapSize1=bitmapLen1+54;
intredress2=h%4;
intbitmapNH2=h*3+redress2;
bitmapLen2=bitmapNH2*bitmapV1;
bitmapSize2=bitmapLen2+54;
memcpy(bitmap2,bitmap,54);
memcpy(bitmap2+2,&bitmapSize2,4);
memcpy(bitmap2+18,&h,4);
doubleHval=(double)bitmapH1/h;
doubleHvala=0;
intHvalb=0,Hvalc=0;
for(i=1;i<=bitmapV1;i++)
{
startH2=bitmapSize2-i*bitmapNH2;
startH1=bitmapSize1-i*bitmapNH1;
B1=bitmap[startH1];
G1=bitmap[startH1+1];
R1=bitmap[startH1+2];
for(j=1;j<=h;j++)
{
startH2=startH2+3;
Hvala=Hvala+Hval;
if(Hvala<1)
{
Hvalb=3;
}
else
{
Hvalc=Hvala;
Hvalb=Hvalc*3;
}
startH1=bitmapSize1-i*bitmapNH1+Hvalb;
B=(bitmap[startH1-3]+B1)/2;
G=(bitmap[startH1-2]+G1)/2;
R=(bitmap[startH1-1]+R1)/2;
bitmap2[startH2-3]=B;
bitmap2[startH2-2]=G;
bitmap2[startH2-1]=R;
B1=bitmap[startH1-3];
G1=bitmap[startH1-2];
R1=bitmap[startH1-1];
}
Hvala=0;
}
}
这一段是对高度进行缩放:
voidBmpSetV(unsignedchar*bitmap,intv,unsignedchar*bitmap2)
{
doubleVval,Vvala=0;
inti,j,Vvalb=0,bitH1,bitNH1,bitV1,redress,bitLen1,bitsize1,bitLen2,bitsize2;
intstart3=0,start2=0,start1=0,G1,B1,R1,R,G,B;
bitH1=*((int*)(bitmap+18));
bitV1=*((int*)(bitmap+22));
redress=bitH1%4;
bitNH1=bitH1*3+redress;
bitLen1=bitNH1*bitV1;
bitsize1=bitLen1+54;
bitLen2=bitNH1*v;
bitsize2=bitLen2+54;
memcpy(bitmap2,bitmap,54);
memcpy(bitmap2+2,&bitsize2,4);
memcpy(bitmap2+22,&v,4);
Vval=bitV1/(double)v;
for(i=1;i<=bitH1;i++)
{
start3=bitsize1-bitNH1+i*3;
B1=bitmap[start3-3];
G1=bitmap[start3-2];
R1=bitmap[start3-1];
for(j=1;j<=v;j++)
{
Vvala=Vvala+Vval;
if(Vvala<1)Vvalb=1;
elseVvalb=Vvala;
start2=bitsize2-j*bitNH1+i*3;
start1=bitsize1-Vvalb*bitNH1+i*3;
if(Vvalb<bitV1)
{
start3=bitsize1-Vvalb*bitNH1-bitNH1+i*3;
if(i==1)
{
bitmap2[start2-3]=bitmap[start1-3];
bitmap2[start2-2]=bitmap[start1-2];
bitmap2[start2-1]=bitmap[start1-1];
}
else
{
B=(B1+bitmap[start1-3])/2;
G=(G1+bitmap[start1-2])/2;
R=(R1+bitmap[start1-1])/2;
bitmap2[start2-3]=B;
bitmap2[start2-2]=G;
bitmap2[start2-1]=R;
B1=bitmap[start3-3];
G1=bitmap[start3-2];
R1=bitmap[start3-1];
}
}
else
{
bitmap2[start2-3]=bitmap[start1-3];
bitmap2[start2-2]=bitmap[start1-2];
bitmap2[start2-1]=bitmap[start1-1];
}
}
Vvala=0;
}
}
示例:
intmain(intargc,char*argv[])
{
unsignedchar*bmp,*bmp2;
bmp=(unsignedchar*)malloc(1244214);
bmp2=(unsignedchar*)malloc(1244214);
FILE*fp=fopen("D:\\开机画面.bmp","rb");
fread(bmp,1,304182,fp);
fclose(fp);
//bmpsetH(bmp,720,bmp2);
//memcpy(bmp,bmp2,1244214);
BmpSetV(bmp,290,bmp2);
fp=fopen("D:\\开机.bmp","wb");
fwrite(bmp2,1,1244214,fp);
fclose(fp);
free(bmp);
free(bmp2);
return0;
}
看到这确实不容易,这里毛票票小编就为大家分享另一个代码
//suofang.cpp:定义控制台应用程序的入口点。
//
#include"stdafx.h"
#include
structheader{
shortBM;
unsignedintfilesize;
unsignedintrecv;
unsignedintoffset;
unsignedintbitmapheadlong;
unsignedintbitmapwith;
unsignedintbitmaphight;
shortbitmappageinfo;
shortbitperpixel;
unsignedintcompress;
unsignedintbitmapsize;
unsignedintlevelresolution;
unsignedintverticalresolution;
unsignedintcolourmap;
unsignedintcolourmapsize;
}bitmapheader;
#if0
typedefstructtagBITMAPFILEHEADER
{
WORDbfType;//位图文件的类型,必须为BM(0-1字节)
DWORDbfSize;//位图文件的大小,以字节为单位(2-5字节)
WORDbfReserved1;//位图文件保留字,必须为0(6-7字节)
WORDbfReserved2;//位图文件保留字,必须为0(8-9字节)
DWORDbfOffBits;//位图数据的起始位置,以相对于位图(10-13字节)
//文件头的偏移量表示,以字节为单位
}BITMAPFILEHEADER;
typedefstructtagBITMAPINFOHEADER{
DWORDbiSize;//本结构所占用字节数(14-17字节)
LONGbiWidth;//位图的宽度,以像素为单位(18-21字节)
LONGbiHeight;//位图的高度,以像素为单位(22-25字节)
WORDbiPlanes;//目标设备的级别,必须为1(26-27字节)
WORDbiBitCount;//每个像素所需的位数,必须是1(双色),(28-29字节)
//4(16色),8(256色)或24(真彩色)之一
DWORDbiCompression;//位图压缩类型,必须是0(不压缩),(30-33字节)
//1(BI_RLE8压缩类型)或2(BI_RLE4压缩类型)之一
DWORDbiSizeImage;//位图的大小,以字节为单位(34-37字节)
LONGbiXPelsPerMeter;//位图水平分辨率,每米像素数(38-41字节)
LONGbiYPelsPerMeter;//位图垂直分辨率,每米像素数(42-45字节)
DWORDbiClrUsed;//位图实际使用的颜色表中的颜色数(46-49字节)
DWORDbiClrImportant;//位图显示过程中重要的颜色数(50-53字节)
}BITMAPINFOHEADER;
typedefstructtagRGBQUAD{
BYTErgbBlue;//蓝色的亮度(值范围为0-255)
BYTErgbGreen;//绿色的亮度(值范围为0-255)
BYTErgbRed;//红色的亮度(值范围为0-255)
BYTErgbReserved;//保留,必须为0
}RGBQUAD;
#endif
voidscale(intsrcwith,intsrcheight,intdestwith,intdestheight)
{
}
int_tmain(intargc,_TCHAR*argv[4000000])
{
BITMAPFILEHEADERbmfHdr;
BITMAPINFOHEADERbi,bi1;
CFilefile("d:\\pic1.bmp",CFile::modeRead);
file.Read(&bmfHdr,sizeof(BITMAPFILEHEADER));
file.Read(&bi,sizeof(BITMAPINFOHEADER));
DWORDdwSize=(bi.biWidth*bi.biBitCount+31)/32*4*bi.biHeight;
PBYTEpBuf=newBYTE[dwSize];
file.Read(pBuf,dwSize);
file.Close();
/////////////////////////////////////////////////////////////////////////
file.Open("d:\\tmp1.bmp",CFile::modeCreate|CFile::modeReadWrite);
memcpy(&bi1,&bi,sizeof(BITMAPINFOHEADER));
bi1.biWidth=4000;//400->200,718->359
bi1.biHeight=4000;//266->133,397->794
DWORDdwSize1=(bi1.biWidth*bi1.biBitCount+31)/32*4*bi1.biHeight;
PBYTEpBuf1=newBYTE[dwSize1];
BYTE*pSrc,*pDest;
//ofstreamfile1("D:\\tmp.log");
#if0
///////////////////////////////////最邻近值采样法(速度快)////////////////////////////////////////////////
floatm_xscale,m_yscale;
m_xscale=(float)bi.biWidth/(float)bi1.biWidth;
m_yscale=(float)bi.biHeight/(float)bi1.biHeight;
unsignedlongk=0;
for(inty=0;y{
for(intx=0;x{
pSrc=pBuf+(int)(y*m_yscale)*bi.biWidth*3+(int)(x*m_xscale)*3;
pDest=pBuf1+(int)(y*bi1.biWidth*3+x*3);
memcpy(pDest,pSrc,3);
//*(unsignedlong*)pDest=k++;
//*(unsignedlong*)pDest=0x000000ff;
}
}
////////////////////////////////////缩放结束///////////////////////////////////////////////
#endif
#if1
/////////////////////////////////////双线性插值法(经过优化)/////////////////////////////////////////////////////
intsw=bi.biWidth-1,sh=bi.biHeight-1,dw=bi1.biWidth-1,dh=bi1.biHeight-1;//源图像宽度,目标图像宽度
intB,N,x,y;//计算出的目标点对应于源图像中的浮点数横坐标N、纵坐标B,目标整数横坐标x、纵坐标y
intnPixelSize=bi.biBitCount/8;//像素大小
BYTE*pLinePrev,*pLineNext;//源图像中的行开始坐标和下一行开始坐标
//BYTE*pDest;
BYTE*pA,*pB,*pC,*pD;//源图像中最邻近的四个点
for(inti=0;i<=dh;++i)//高度递增
{
pDest=(BYTE*)(pBuf1+bi1.biWidth*i*nPixelSize);
y=i*sh/dh;
N=dh-i*sh%dh;
pLinePrev=(BYTE*)(pBuf+bi.biWidth*y*nPixelSize);
y++;
pLineNext=(N==dh)?pLinePrev:(BYTE*)(pBuf+bi.biWidth*y*nPixelSize);
for(intj=0;j<=dw;++j)//宽度递增
{
x=j*sw/dw*nPixelSize;
B=dw-j*sw%dw;
pA=pLinePrev+x;
pB=pA+nPixelSize;
pC=pLineNext+x;
pD=pC+nPixelSize;
if(B==dw)
{
pB=pA;
pD=pC;
}
for(intk=0;k<nPixelSize;++k)
*pDest++=(BYTE)(int)(
(B*N*(*pA++-*pB-*pC+*pD)+dw*N**pB++
+dh*B**pC+++(dw*dh-dh*B-dw*N)**pD++
+dw*dh/2)/(double)(dw*dh)
);
}
}
#endif
file.Write(&bmfHdr,sizeof(BITMAPFILEHEADER));
file.Write(&bi1,sizeof(BITMAPINFOHEADER));
file.Write(pBuf1,dwSize1);
file.Close();
//==releasepmemory==
deletepBuf;
pBuf=NULL;
deletepBuf1;
pBuf1=NULL;
//==releasepmemory===
if(pDest!=NULL)
{
pDest=NULL;
}
//if(pSrc!=NULL)
//{
//pSrc=NULL;
//}
//////////////////////////////////////////////////////////////
//MessageBox("Testsuccessful","Info",MB_OKCANCEL);
return0;
}