摘要:可移植像素圖格式,灰度圖格式,位圖格式的介紹簡介可移植像素圖格式,可移植灰度圖格式和可移植位圖格式是便于跨平臺的圖像格式。每個文件的開頭兩個字節碼作為文件描述符,指出具體格式和編碼形式。
可移植像素圖格式 PPM,灰度圖格式 PGM,位圖格式 PBM 的介紹 簡介
可移植像素圖格式(PPM),可移植灰度圖格式(PGM)和可移植位圖格式(PBM)是便于跨平臺的圖像格式。有時候也被統稱為 PNM 格式
文件格式描述這三種格式其實是一樣的描述方法,只不過 PBM 是單色,PGM 是灰度圖,PPM 使用 RGB 顏色。
每個文件的開頭兩個字節(ASCII 碼)作為文件描述符,指出具體格式和編碼形式。
Type | Magic number | Extension | Colors | |
---|---|---|---|---|
ASCII | Binary | |||
Portable BitMap | P1 | P4 | .pbm | 0–1 (white & black) |
Portable GrayMap | P2 | P5 | .pgm | 0–255 (gray scale) |
Portable PixMap | P3 | P6 | .ppm | 0–255 (RGB) |
Portable Arbitrary Map | P7 | .pam | 0–255 (RGB_ALPHA) |
注意每行結束有換行符
P1 # This is an example bitmap of the letter "J" 6 10 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 1 0 0 0 0 0 1 0 1 0 0 0 1 0 0 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0
上面的圖像是一個J
PGMP2 # Shows the word "FEEP" (example from Netpbm man page on PGM) 24 7 15 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 3 3 3 3 0 0 7 7 7 7 0 0 11 11 11 11 0 0 15 15 15 15 0 0 3 0 0 0 0 0 7 0 0 0 0 0 11 0 0 0 0 0 15 0 0 15 0 0 3 3 3 0 0 0 7 7 7 0 0 0 11 11 11 0 0 0 15 15 15 15 0 0 3 0 0 0 0 0 7 0 0 0 0 0 11 0 0 0 0 0 15 0 0 0 0 0 3 0 0 0 0 0 7 7 7 7 0 0 11 11 11 11 0 0 15 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0
上面的圖像是
P3 3 2 255 # The part above is the header # "P3" means this is a RGB color image in ASCII # "3 2" is the width and height of the image in pixels # "255" is the maximum value for each color # The part below is image data: RGB triplets 255 0 0 0 255 0 0 0 255 255 255 0 255 255 255 0 0 0
把上面六個像素放大后顯示如下
P7 WIDTH 4 HEIGHT 2 DEPTH 4 MAXVAL 255 TUPLTYPE RGB_ALPHA ENDHDR 0000FFFF 00FF00FF FF0000FF FFFFFFFF 0000FF7F 00FF007F FF00007F FFFFFF7F
上面數據放大顯示如下:
下面是簡單的ppm包
package ppm import ( "fmt" "io" "os" ) // WriteTo outputs 8-bit P6 PPM format to an io.Writer. func (b *Bitmap) WritePpmTo(w io.Writer) (err error) { // magic number if _, err = fmt.Fprintln(w, "P6"); err != nil { return } // comments for _, c := range b.Comments { if _, err = fmt.Fprintln(w, c); err != nil { return } } // x, y, depth _, err = fmt.Fprintf(w, "%d %d 255 ", b.cols, b.rows) if err != nil { return } // raster data in a single write b3 := make([]byte, 3*len(b.px)) n1 := 0 for _, px := range b.px { b3[n1] = px.R b3[n1+1] = px.G b3[n1+2] = px.B n1 += 3 } if _, err = w.Write(b3); err != nil { return } return } // WriteFile writes to the specified filename. func (b *Bitmap) WritePpmFile(fn string) (err error) { var f *os.File if f, err = os.Create(fn); err != nil { return } if err = b.WritePpmTo(f); err != nil { return } return f.Close() }
下面是生成ppm的程序
package main // Files required to build supporting package raster are found in: // * This task (immediately above) // * Bitmap task import ( "raster" "fmt" ) func main() { b := raster.NewBitmap(400, 300) b.FillRgb(0x240008) // a dark red err := b.WritePpmFile("write.ppm") if err != nil { fmt.Println(err) } }用C生成PPM文件
#includeint main() { const char *filename = "n.pgm"; int x, y; /* size of the image */ const int x_max = 100; /* width */ const int y_max = 100; /* height */ /* 2D array for colors (shades of gray) */ unsigned char data[y_max][x_max]; /* color component is coded from 0 to 255 ; it is 8 bit color file */ const int MaxColorComponentValue = 255; FILE * fp; /* comment should start with # */ const char *comment = "# this is my new binary pgm file"; /* fill the data array */ for (y = 0; y < y_max; ++y) { for (x = 0; x < x_max; ++x) { data[y][x] = (x + y) & 255; } } /* write the whole data array to ppm file in one step */ /* create new file, give it a name and open it in binary mode */ fp = fopen(filename, "wb"); /* write header to the file */ fprintf(fp, "P5 %s %d %d %d ", comment, x_max, y_max, MaxColorComponentValue); /* write image data bytes to the file */ fwrite(data, sizeof(data), 1, fp); fclose(fp); printf("OK - file %s saved ", filename); return 0; }
或者
imglib.h
#ifndef _IMGLIB_0 #define _IMGLIB_0 #include#include #include #include #include #include typedef unsigned char color_component; typedef color_component pixel[3]; typedef struct { unsigned int width; unsigned int height; pixel * buf; } image_t; typedef image_t * image; image alloc_img(unsigned int width, unsigned int height); void free_img(image); void fill_img(image img, color_component r, color_component g, color_component b ); void put_pixel_unsafe( image img, unsigned int x, unsigned int y, color_component r, color_component g, color_component b ); void put_pixel_clip( image img, unsigned int x, unsigned int y, color_component r, color_component g, color_component b ); #define GET_PIXEL(IMG, X, Y) (IMG->buf[ ((Y) * IMG->width + (X)) ]) #endif
imglib.c
image alloc_img(unsigned int width, unsigned int height) { image img; img = malloc(sizeof(image_t)); img->buf = malloc(width * height * sizeof(pixel)); img->width = width; img->height = height; return img; } void free_img(image img) { free(img->buf); free(img); } void fill_img( image img, color_component r, color_component g, color_component b ) { unsigned int i, n; n = img->width * img->height; for (i=0; i < n; ++i) { img->buf[i][0] = r; img->buf[i][1] = g; img->buf[i][2] = b; } } void put_pixel_unsafe( image img, unsigned int x, unsigned int y, color_component r, color_component g, color_component b ) { unsigned int ofs; ofs = (y * img->width) + x; img->buf[ofs][0] = r; img->buf[ofs][1] = g; img->buf[ofs][2] = b; } void put_pixel_clip( image img, unsigned int x, unsigned int y, color_component r, color_component g, color_component b ) { if (x < img->width && y < img->height) put_pixel_unsafe(img, x, y, r, g, b); }
output_ppm
#include "imglib.h" void output_ppm(FILE *fd, image img) { unsigned int n; (void) fprintf(fd, "P6 %d %d 255 ", img->width, img->height); n = img->width * img->height; (void) fwrite(img->buf, sizeof(pixel), n, fd); (void) fflush(fd); }生成raw rgb格式
其實raw rgb和ppm數據是一樣的,我們生成出300幀的ppm數據,然后送給ffplayer(ffmpeg)來播放
下面是我的go代碼:
// // Created by : Harris Zhu // Filename : genrgb.go // Author : Harris Zhu // Created On : 2018-09-14 02:13:11 // Last Modified : 2018-09-14 02:13:11 // Update Count : 1 // Tags : // Description : // Conclusion : // //======================================================================= package main import ( "bufio" "os" "strconv" "time" "github.com/urfave/cli" ) func main() { app := cli.NewApp() app.Name = "genrgb" app.Version = "1.0.0" app.Compiled = time.Now() app.Authors = []cli.Author{ cli.Author{ Name: "Harris Zhu", Email: "zhuzhzh@163.com", }, } app.Usage = "svpaser" name := "rgb.data" w := 600 h := 480 f := 20 app.Action = func(c *cli.Context) error { name = c.Args().Get(0) w, _ = strconv.Atoi(c.Args().Get(1)) h, _ = strconv.Atoi(c.Args().Get(2)) f, _ = strconv.Atoi(c.Args().Get(3)) genrgb(name, w, h, f) return nil } app.Run(os.Args) } func genrgb(filepath string, w int, h int, f int) { fout, err := os.Create(filepath) defer fout.Close() if err != nil { panic(err) } bufout := bufio.NewWriter(fout) // bufout.WriteString("P6 # this is my ppm file ") // bufout.WriteString(strconv.Itoa(w)) // bufout.WriteString(" ") // bufout.WriteString(strconv.Itoa(h)) // bufout.WriteString(" ") // bufout.WriteString(strconv.Itoa(255)) // bufout.WriteString(" ") r := []byte{255, 0, 0} g := []byte{0, 255, 0} b := []byte{0, 0, 255} dcnt := 0 //k: frame //i: height column //j: width line for k := 0; k < f; k++ { for i := 0; i < h; i++ { //fmt.Println("hline: ", i) for j := 0; j < w; j++ { if k%3 == 0 { bufout.Write(r) dcnt++ } else if k%3 == 1 { bufout.Write(g) dcnt++ } else if k%3 == 2 { bufout.Write(b) dcnt++ } //fmt.Println("vline: ", j) } } bufout.Flush() //fmt.Printf("total pixels = %d ", dcnt) } }
下面是makefile:
FILE = genrgb GENFILE = rgb.data b build: go build -gcflags "-N -l" $(FILE).go g gen: ./$(FILE) $(GENFILE) 60 40 300 p play: cat $(GENFILE) | ffplay -i pipe:0 -f rawvideo -pix_fmt rgb24 -video_size 60x40
依次執行make b; make g; make p可以看到下面的播放視頻
它交替顯示紅綠藍
總結RGB比較直觀的圖像顯示方法,但相對YUV來說比較占空間,每個像素占用3個byte, 像1080P的圖像就要占用$1920*1080*3=6,220,800byte$, 但它用于作為圖像和顯示入門是非常好的。能夠讓你很快就常會編程顯示圖像。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/11071.html
摘要:因為分別代表不同顏色信號,所以直接使用與信號表示色度的。彩色圖像記錄的格式,常見的有等。主要的抽樣格式有和。的表示法稱為表示法表示完全取樣。表示的水平取樣,垂直采樣。格式則有一通道。 YUV和RGB詳解 前言 YUV,是一種顏色編碼方法。常使用在各個視頻處理組件中。 YUV在對照片或視頻編碼時,考慮到人類的感知能力,允許降低色度的帶寬。 YUV是編碼true-color時使用的顏色空...
摘要:了解華為海思的方案海思的前身是華為的半導體部門,主要產品線包括智能手機處理器麒麟系列,視頻采集和編解碼處理器系列,無線通信方向芯片等。 目錄 一、視頻行業1、視頻...
閱讀 3376·2021-11-22 09:34
閱讀 2881·2021-10-09 09:43
閱讀 1461·2021-09-24 09:47
閱讀 2210·2019-08-30 12:53
閱讀 1008·2019-08-29 14:00
閱讀 3369·2019-08-29 13:17
閱讀 2277·2019-08-28 18:00
閱讀 1294·2019-08-26 12:00