C++保存Bitmap圖片,關(guān)于位圖數(shù)據(jù)怎么4字節(jié)對(duì)齊
關(guān)于位圖數(shù)據(jù)怎么4字節(jié)對(duì)齊:
假如一張2*2的RGB圖片如下:
那么它在內(nèi)存中數(shù)據(jù)理論應(yīng)該為(圖片數(shù)據(jù)在內(nèi)存中是連續(xù)的,也就是說(shuō)一行數(shù)據(jù)緊接著另一行數(shù)據(jù),這里為了直觀所以布局成這樣)
(0xFF 0x00 0x00) (0x00 0xFF 0x00)
(0x00 0x00 0x00) (0xFF 0xFF 0xFF)
但是位圖保存到磁盤時(shí)規(guī)定圖片的數(shù)據(jù)每行的字節(jié)數(shù)要能被4整除(具體什么的我也不太清楚,對(duì)于我們一般人來(lái)說(shuō),我認(rèn)為只要知道有這規(guī)定就行),對(duì)于上面的這張圖片它每行數(shù)據(jù)為6個(gè)字節(jié),顯然6除以4還于2,顯然沒(méi)有4字節(jié)對(duì)齊,所以需要在每行數(shù)據(jù)后面補(bǔ)充2個(gè)字節(jié)的數(shù)據(jù)(隨便值為多少,只要時(shí)2字節(jié)就行),因此補(bǔ)充后上圖的數(shù)據(jù)變成:
(0xFF 0x00 0x00) (0x00 0xFF 0x00) (0x00 0x00)
(0x00 0x00 0x00) (0xFF 0xFF 0xFF)(0x00 0x00)
這樣子的話將圖片保存到磁盤中時(shí)就不會(huì)出現(xiàn)問(wèn)題了。(不管圖片每個(gè)像素是8字節(jié) 16字節(jié) 還是24字節(jié)都一樣)
在實(shí)際中怎么操作呢,假如我們從攝像頭中取出一張2*2的圖片如下:
其數(shù)據(jù)在bitmapData 中,一共6個(gè)字節(jié)
當(dāng)我們想要將它保存到磁盤中去時(shí),我們可以調(diào)用下面這個(gè)函數(shù)simple_4ByteAlignment(2,2,bitmapData)(一些說(shuō)明全在寫(xiě)在下面函數(shù)中。)
bool?simple_4ByteAlignment(int?width,?int?height,?unsigned?char?*?bitmapData) { if?(!bitmapData) { return?false; } unsigned?char?*?fData=0; //length?數(shù)據(jù)的總長(zhǎng)度 //fillCount?每行需要補(bǔ)充的字節(jié)數(shù) int?length=0,?fillCount=0; length?=?width*3;//這里乘3是因?yàn)槲覀儓D片為Rgb圖片,每個(gè)像素占3個(gè)字節(jié) if?(length?%?4?!=?0)//先計(jì)算補(bǔ)充字節(jié)后,數(shù)據(jù)的總字節(jié)數(shù) { fillCount?=?4?-?length?%?4; length?+=?fillCount; } length?*=height; //分配相應(yīng)長(zhǎng)度的內(nèi)存 fData?=?(uchar*)malloc(length); if?(!fData) { return?false; } memset(fData,?0,?length); for?(size_t?i?=?0;?i?<?height;?i++) { //由于需要4字節(jié)對(duì)齊,所以需要對(duì)每行進(jìn)行補(bǔ)充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); //釋放內(nèi)存?? free(fData); return?ret; }
下面兩個(gè)為保存圖片為bmp格式的函數(shù),一個(gè)保存圖片為rgb888一個(gè)保存為rgb565。
//根據(jù)bitmapData的(RGB)數(shù)據(jù),保存bitmap?? //filename是要保存到物理硬盤的文件名(包括路徑)?? //dateSize?表示圖形數(shù)據(jù)的大小 //bitmapData?數(shù)據(jù)必須經(jīng)過(guò)4字節(jié)對(duì)齊 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固定為這個(gè) 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)過(guò)4字節(jié)對(duì)齊 static?BOOL?WriteBitmap565ToFile(const?char?*?filename,?int?width,?int?height,?unsigned?char?*?bitmapData,?int?dateSize) { BITMAPFILEHEADER?bitmapFileHeader; //填充BITMAPINFOHEADER?? BITMAPINFOHEADER?bitmapInfoHeader; RGBQUAD?bmiColors[3];?//定義調(diào)色板 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; }