關于位圖數(shù)據(jù)怎么4字節(jié)對齊:
假如一張2*2的RGB圖片如下:
那么它在內存中數(shù)據(jù)理論應該為(圖片數(shù)據(jù)在內存中是連續(xù)的,也就是說一行數(shù)據(jù)緊接著另一行數(shù)據(jù),這里為了直觀所以布局成這樣)
(0xFF 0x00 0x00) (0x00 0xFF 0x00)
(0x00 0x00 0x00) (0xFF 0xFF 0xFF)
但是位圖保存到磁盤時規(guī)定圖片的數(shù)據(jù)每行的字節(jié)數(shù)要能被4整除(具體什么的我也不太清楚,對于我們一般人來說,我認為只要知道有這規(guī)定就行),對于上面的這張圖片它每行數(shù)據(jù)為6個字節(jié),顯然6除以4還于2,顯然沒有4字節(jié)對齊,所以需要在每行數(shù)據(jù)后面補充2個字節(jié)的數(shù)據(jù)(隨便值為多少,只要時2字節(jié)就行),因此補充后上圖的數(shù)據(jù)變成:
(0xFF 0x00 0x00) (0x00 0xFF 0x00) (0x00 0x00)
(0x00 0x00 0x00) (0xFF 0xFF 0xFF)(0x00 0x00)
這樣子的話將圖片保存到磁盤中時就不會出現(xiàn)問題了。(不管圖片每個像素是8字節(jié) 16字節(jié) 還是24字節(jié)都一樣)
在實際中怎么操作呢,假如我們從攝像頭中取出一張2*2的圖片如下:
其數(shù)據(jù)在bitmapData 中,一共6個字節(jié)
當我們想要將它保存到磁盤中去時,我們可以調用下面這個函數(shù)simple_4ByteAlignment(2,2,bitmapData)(一些說明全在寫在下面函數(shù)中。)
bool?simple_4ByteAlignment(int?width,?int?height,?unsigned?char?*?bitmapData)
{
if?(!bitmapData)
{
return?false;
}
unsigned?char?*?fData=0;
//length?數(shù)據(jù)的總長度
//fillCount?每行需要補充的字節(jié)數(shù)
int?length=0,?fillCount=0;
length?=?width*3;//這里乘3是因為我們圖片為Rgb圖片,每個像素占3個字節(jié)
if?(length?%?4?!=?0)//先計算補充字節(jié)后,數(shù)據(jù)的總字節(jié)數(shù)
{
fillCount?=?4?-?length?%?4;
length?+=?fillCount;
}
length?*=height;
//分配相應長度的內存
fData?=?(uchar*)malloc(length);
if?(!fData)
{
return?false;
}
memset(fData,?0,?length);
for?(size_t?i?=?0;?i?<?height;?i++)
{
//由于需要4字節(jié)對齊,所以需要對每行進行補充fillCount的字節(jié)的數(shù)據(jù)。
memcpy(fData+(width*3+?fillCount?)*i,?bitmapData+width?*?3*i,?width?*?3);
}
bool?ret=WriteBitmap888ToFile("你要保存的圖片的路徑.bmp",?width,?height,?(uchar*)fData,?length);
//釋放內存??
free(fData);
return?ret;
}下面兩個為保存圖片為bmp格式的函數(shù),一個保存圖片為rgb888一個保存為rgb565。
//根據(jù)bitmapData的(RGB)數(shù)據(jù),保存bitmap??
//filename是要保存到物理硬盤的文件名(包括路徑)??
//dateSize?表示圖形數(shù)據(jù)的大小
//bitmapData?數(shù)據(jù)必須經(jīng)過4字節(jié)對齊
static?BOOL?WriteBitmap888ToFile(const?char?*?filename,?int?width,?int?height,?unsigned?char?*?bitmapData,?int?dateSize)
{
//填充BITMAPINFOHEADER??
BITMAPINFOHEADER?bitmapInfoHeader;
memset(&bitmapInfoHeader,?0,?sizeof(BITMAPINFOHEADER));
bitmapInfoHeader.biSize?=?sizeof(BITMAPINFOHEADER);
bitmapInfoHeader.biWidth?=?width;
bitmapInfoHeader.biHeight?=?height;
bitmapInfoHeader.biPlanes?=?1;
bitmapInfoHeader.biBitCount?=?24;
bitmapInfoHeader.biCompression?=?BI_RGB;
bitmapInfoHeader.biSizeImage?=?dateSize;
//填充BITMAPFILEHEADER??
BITMAPFILEHEADER?bitmapFileHeader;
memset(&bitmapFileHeader,?0,?sizeof(BITMAPFILEHEADER));
bitmapFileHeader.bfType?=?0x4d42;???//BM固定為這個
bitmapFileHeader.bfOffBits?=?sizeof(BITMAPFILEHEADER)?+?sizeof(BITMAPINFOHEADER);
bitmapFileHeader.bfSize?=?bitmapFileHeader.bfOffBits?+?dateSize;
bitmapFileHeader.bfReserved1?=?0;
bitmapFileHeader.bfReserved2?=?0;
FILE?*?filePtr?=?0;???????
fopen_s(&filePtr,?filename,?"wb");
if?(NULL?==?filePtr)
{
return?FALSE;
}
fwrite(&bitmapFileHeader,?sizeof(BITMAPFILEHEADER),?1,?filePtr);
fwrite(&bitmapInfoHeader,?sizeof(BITMAPINFOHEADER),?1,?filePtr);
fwrite(bitmapData,?bitmapInfoHeader.biSizeImage,?1,?filePtr);
fclose(filePtr);
return?TRUE;
}
//根據(jù)bitmapData的(RGB)數(shù)據(jù),保存bitmap??
//filename是要保存到物理硬盤的文件名(包括路徑)??
//dateSize?表示圖形數(shù)據(jù)的大小,
//bitmapData?數(shù)據(jù)必須經(jīng)過4字節(jié)對齊
static?BOOL?WriteBitmap565ToFile(const?char?*?filename,?int?width,?int?height,?unsigned?char?*?bitmapData,?int?dateSize)
{
BITMAPFILEHEADER?bitmapFileHeader;
//填充BITMAPINFOHEADER??
BITMAPINFOHEADER?bitmapInfoHeader;
RGBQUAD?bmiColors[3];?//定義調色板
bitmapFileHeader.bfType?=?0x4d42;???//"BM"
bitmapFileHeader.bfOffBits?=?sizeof(BITMAPFILEHEADER)?+?sizeof(BITMAPINFOHEADER)?+?sizeof(RGBQUAD)?*?3;?
bitmapFileHeader.bfSize?=?bitmapFileHeader.bfOffBits?+?dateSize;
bitmapFileHeader.bfReserved1?=?0;
bitmapFileHeader.bfReserved2?=?0;
memset(&bitmapInfoHeader,?0,?sizeof(BITMAPINFOHEADER));
bitmapInfoHeader.biSize?=?sizeof(BITMAPINFOHEADER);
bitmapInfoHeader.biWidth?=?width;
bitmapInfoHeader.biHeight?=?-height;
bitmapInfoHeader.biPlanes?=?1;
bitmapInfoHeader.biBitCount?=?16;
bitmapInfoHeader.biCompression?=?BI_BITFIELDS;
bitmapInfoHeader.biSizeImage?=?dateSize;
bmiColors[0].rgbBlue?=?0;
bmiColors[0].rgbGreen?=?0xF8;
bmiColors[0].rgbRed?=?0;
bmiColors[0].rgbReserved?=?0;
bmiColors[1].rgbBlue?=?0xE0;
bmiColors[1].rgbGreen?=?0x07;
bmiColors[1].rgbRed?=?0;
bmiColors[1].rgbReserved?=?0;
bmiColors[2].rgbBlue?=?0x1F;
bmiColors[2].rgbGreen?=?0;
bmiColors[2].rgbRed?=?0;
bmiColors[2].rgbReserved?=?0;
FILE?*?filePtr?=?0;???????
fopen_s(&filePtr,?filename,?"wb");
if?(NULL?==?filePtr)
{
return?FALSE;
}
fwrite(&bitmapFileHeader,?sizeof(BITMAPFILEHEADER),?1,?filePtr);
fwrite(&bitmapInfoHeader,?sizeof(BITMAPINFOHEADER),?1,?filePtr);
fwrite(bmiColors,3?*?sizeof(RGBQUAD),?1,?filePtr);
fwrite(bitmapData,?bitmapInfoHeader.biSizeImage,?1,?filePtr);
fclose(filePtr);
return?TRUE;
}




