工作上需要測試視頻解碼之后的數(shù)據(jù)或是經(jīng)過算法處理之后圖像是否符合預(yù)期,可以使用一些現(xiàn)有的格式比如 JPG、PNG等常見格式。因?yàn)橹皇桥R時(shí)需要我們可以使用一種更簡單的圖像格式 PPM。PPM是一種簡單的圖片格式,我們可以通過PPM進(jìn)行圖像相關(guān)的學(xué)習(xí)。
PPM格式介紹:
PPM圖像格式分為兩部分,分別為頭部分和圖像數(shù)據(jù)部分。
頭部分:
由3部分組成,通過換行或空格進(jìn)行分割,一般PPM的標(biāo)準(zhǔn)是空格。 第1部分:P3或P6,指明PPM的編碼格式, 第2部分:圖像的寬度和高度,通過ASCII表示, 第3部分:最大像素值,0-255字節(jié)表示。
每個(gè)圖像文件的開頭都通過2個(gè)字節(jié)「magic number」來表明文件格式的類型(PBM, PGM, PPM),以及編碼方式(ASCII 或 Binary),magic number分別為P1、P2、P3、P4、P5、P。
| Magic Number | Type | Encoding |
|---|---|---|
| P1 | Bitmap | ASCII |
| P2 | Graymap | ASCII |
| P3 | Pixmap | ASCII |
| P4 | Bitmap | Binary |
| P5 | Graymap | Binary |
| P6 | Pixmap | Binary |
圖像數(shù)據(jù)部分:
ASCII格式:按RGB的順序排列,RGB中間用空格隔開,圖片每一行用回車隔開。 Binary格式:PPM用24bits代表每一個(gè)像素,紅綠藍(lán)分別占用8bits。
PPM文件示例代碼:
C\C++:
static int save_rgba_as_ppm(unsigned char* p_rgba, int width, int height, const char *filepath)
{
FILE *pFile;
pFile = fopen(filepath, "wb");
if (pFile == NULL)
return -1;
// Write header
// 默認(rèn)是 P6 顏色像素 binary 版本
fprintf(pFile, "P6\n%d %d\n255\n", width, height);
// Write pixel data
for (int i = width*height; i > 0; i--)
{
fwrite(pRGBA, 1, 1, pFile);
fwrite(pRGBA + 1, 1, 1, pFile);
fwrite(pRGBA + 2, 1, 1, pFile);
pRGBA += 4;
}
// Close file
fclose(pFile);
return 0;
}
Java:
public void save_rgba_as_ppm(byte[] rgba, int w, int h, String filepath) {
try {
FileOutputStream fileOutputStream = new FileOutputStream(new File(filepath));
BufferedOutputStream outputStream = new BufferedOutputStream(fileOutputStream);
String s = String.format("P6\n%d %d\n255\n", w, h);
outputStream.write(s.getBytes());
int offset = 0;
for (int i = w*h; i > 0; i--)
{
outputStream.write(rgba[offset]);
outputStream.write(rgba[offset + 1]);
outputStream.write(rgba[offset + 2]);
offset += 4;
}
outputStream.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
總結(jié):
使用PPM可以給圖像處理相關(guān)的調(diào)試工作帶來便利。沒有復(fù)雜的壓縮計(jì)算,原始流數(shù)據(jù)存儲。希望能幫助到從事相關(guān)工作的童鞋。:)