音视频入门-18-手动生成一张GIF图片

* 音视频入门文章目录 *

GIF 编码知识

1.png

GIF 包含的数据块:

  • 文件头(Header)

  • 逻辑屏幕标识符(Logical Screen Descriptor)

  • 全局颜色表(Global Color Table)

  • Application Extension

  • Comment Extension

  • 图形控制扩展(Graphic Control Extension)

  • 图像标识符(Image Descriptor)

  • 局部颜色表(Local Color Table)

  • 基于颜色表的图像数据(Image Data)

  • Plain Text Extension

  • 文件结尾(Trailer)

GIF 编码步骤

今天的目标是做出一张尺寸 700x700、7 个颜色画面切换的 GIF 动画。

2.gif

文件头(Header)

GIF 的前 6 个字节内容是 GIF 的署名和版本号。有两个版本 GIF87a GIF89aGIF89a 版本才有多帧动画,所有这里使用 89a 版本。

示例代码:

// GIF 文件头,6 个字节内容是 GIF 的署名和版本号
uint8_t gif_header[] = {0x47, 0x49, 0x46, 0x38, 0x39, 0x61};
fwrite(gif_header, 6, 1, gif_file);

逻辑屏幕标识符(Logical Screen Descriptor)

从上一篇 音视频入门-17-GIF文件格式详解 我们知道:

逻辑屏幕标识符(7 个字节):

  • 屏幕逻辑宽度:2 字节;

  • 屏幕逻辑高度:2 字节;

  • 打包值,大小为 1 字节

    • m - 全局颜色表标志,1 bit;
    • cr - 颜色深度,3 bit;(x: 可忽略)
    • s - 分类标志, 1 bit; (x: 不使用,设为 0)
    • pixel - 全局颜色列表大小,3 bit;
  • 背景颜色索引: 1 字节;

  • 像素宽高比: 1 字节;(x: 不使用,设为 0)

示例代码:

// 逻辑屏幕标识符
uint16_t gif_width = 700;
uint16_t gif_height = 700;
// 0xF2 = 1   1 1 1   0   0 1 0
uint8_t gif_logical_screen_pack_byte = 0xF2;
uint8_t gif_bg_color_index = 0;
uint8_t gif_pixel_aspect = 0;

fputc(gif_width >> 0, gif_file); // width low 8
fputc(gif_width >> 8, gif_file); // width high 8
fputc(gif_height  >> 0, gif_file); // height low 8
fputc(gif_height  >> 8, gif_file); // height high 8
fputc(gif_logical_screen_pack_byte, gif_file);
fputc(gif_bg_color_index, gif_file);
fputc(gif_pixel_aspect, gif_file);

全局颜色表(Global Color Table)

每个颜色索引由三字节组成,按 RGB 顺序排列。

由 【逻辑屏幕标识符】可知,颜色的索引数(2^(pixel+1))是 2 的倍数,如果图片颜色数目不够要补足。

比如,我们的图片用了 7 个颜色,颜色索引数是 8,所以最后再加一个颜色(占位,不使用)。

示例代码:

// 颜色表
uint32_t rainbowColors[] = {
        0XFF0000, // 赤
        0XFFA500, // 橙
        0XFFFF00, // 黄
        0X00FF00, // 绿
        0X007FFF, // 青
        0X0000FF, // 蓝
        0X8B00FF, // 紫
        0X000000  // 黑
};
// 全局颜色表、
for(int i = 0; i < 8; i++) {
    // 根据颜色索引取出颜色表中的颜色
    uint32_t color_rgb = rainbowColors[i];
    // 当前颜色 R 分量
    uint8_t R = (color_rgb & 0xFF0000) >> 16;
    // 当前颜色 G 分量
    uint8_t G = (color_rgb & 0x00FF00) >> 8;
    // 当前颜色 B 分量
    uint8_t B = color_rgb & 0x0000FF;
    fputc(R, gif_file);
    fputc(G, gif_file);
    fputc(B, gif_file);
}

Application Extension

Application Extension 这 19 个字节基本上 GIF 都一样。

0x21, 0xFF, 0x0B, 0x4E, 0x45, 0x54, 0x53, 0x43, 0x41, 0x50, 0x45, 0x32, 0x2E, 0x30, 0x03, 0x01, 0x00, 0x00, 0x00

代表的内容是 NETSCAPE2.0

示例代码:

// Application Extension
uint8_t gif_application_extension[] = {0x21, 0xFF, 0x0B, 0x4E, 0x45, 0x54, 0x53, 0x43, 0x41, 0x50, 0x45, 0x32, 0x2E, 0x30, 0x03, 0x01, 0x00, 0x00, 0x00};
fwrite(gif_application_extension, 19, 1, gif_file);

Comment Extension

这里允许你将 ASCII 文本嵌入到 GIF 文件,有时被用来图像描述、图像信贷或其他人类可读的元数据,如图像捕获的 GPS 定位。

0x21, 0xFE, 0x20, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x20, 0x77, 0x69, 0x74, 0x68, 0x20, 0x65, 0x7A, 0x67, 0x69, 0x66, 0x2E, 0x63, 0x6F, 0x6D, 0x20, 0x47, 0x49, 0x46, 0x20, 0x6D, 0x61, 0x6B, 0x65, 0x72, 0x00

代表的内容是 Created with ezgif.com GIF maker

示例代码:

// Comment Extension
// Created with ezgif.com GIF maker
uint8_t gif_comment_extension[] = {0x21, 0xFE, 0x20, 0x43, 0x72, 0x65, 0x61, 0x74, 0x65, 0x64, 0x20, 0x77, 0x69, 0x74, 0x68, 0x20, 0x65, 0x7A, 0x67, 0x69, 0x66, 0x2E, 0x63, 0x6F, 0x6D, 0x20, 0x47, 0x49, 0x46, 0x20, 0x6D, 0x61, 0x6B, 0x65, 0x72, 0x00};
fwrite(gif_comment_extension, 36, 1, gif_file);

图形控制扩展(Graphic Control Extension)

我们的 GIF 不使用处置方法 不使用透明色 图像延迟 50

所以,这里就是 0x21, 0xF9, 0x04, 0x00, 0x32, 0x00, 0xFF, 0x00

示例代码:

// 图形控制扩展
uint8_t gif_graphic_control_extension[] = {0x21, 0xF9, 0x04, 0x00, 0x32, 0x00, 0xFF, 0x00};
fwrite(gif_graphic_control_extension, 8, 1, gif_file);

图像标识符(Image Descriptor)

我们的 GIF 没有局部颜色表 顺序排列 局部颜色表大小为 0

所以,这里就是 0x2C, 0x00, 0x00, 0x00, 0x00, 0xBC, 0x02, 0xBC, 0x02, 0x00

示例代码:

// 图像标识符
uint8_t gif_image_descriptor[] = {0x2C, 0x00, 0x00, 0x00, 0x00, 0xBC, 0x02, 0xBC, 0x02, 0x00};
fwrite(gif_image_descriptor, 10, 1, gif_file);

局部颜色表(Local Color Table)

如果有局部颜色表,则跟 全局颜色表(Global Color Table) 一样的格式。

基于颜色表的图像数据(Image Data)

这里是最关键的图像数据,生成步骤如下:

  • 1.根据全局颜色表或者局部颜色表,生成一张图像的颜色索引数据
  • 2.使用 LZW 算法压缩上一步生成的数据
  • 3.将压缩后的数据按照格式写入文件

1.生成索引数据

我们要生成的 GIF 尺寸 700x700,有 7 张图像,每张图像一个颜色 绿

颜色已经写入全局颜色表中;

每个颜色索引 1 字节;

示例代码:

// 基于颜色表的图像数据
uint8_t *gif_one_frame_raw = malloc(700 * 700);
memset(gif_one_frame_raw, i, 700*700);

2.LZW 压缩数据

LZW 压缩算法不在本次研究范围,直接用即可。

//  GIF 一帧图像的数据压缩后大小
unsigned long compressed_size;
// GIF 一帧图像的数据解压后的数据
unsigned char *img;
lzw_compress_gif(
        3,
        700*700,
        gif_one_frame_raw,
        &compressed_size,
        &img
);

3.按照格式写入文件

第一个字节表示 LZW 编码初始表大小的位数,用于使用 LZW 算法解压数据。

后面的是图像数据块:

每个数据块第一个字节表示数据块大小(不包括这个字节)
数据块后面的一个字节表示后续数据块大小
当数据块后面的一个字节是 0 ,表示数据结束了

[图片上传失败...(image-d75d16-1607049927310)]

示例代码:

fputc(0x03, gif_file);
unsigned long current_index = 0;
while (current_index < compressed_size) {
    if((current_index + 0xFF) >= compressed_size) {
        unsigned long diff = compressed_size - current_index;
        fputc(diff, gif_file);
        fwrite(img+current_index, diff, 1, gif_file);
        fputc(0x00, gif_file);
        current_index += diff;
    } else {
        fputc(0xFF, gif_file);
        fwrite(img+current_index, 0xFF, 1, gif_file);
        current_index += 0xFF;
    }
}

Plain Text Extension

这个特性不起作用; 浏览器和图片处理应用程序,如 Photoshop 忽略它, GIFLIB 并不试图解释它。

所以直接忽略。

文件结尾(Trailer)

标识 GIF 文件结束,固定值 0x3B。

当解析程序读到 0x3B 时,文件终结。

示例代码:

// GIF 文件结束: 0x3B
fputc(0x3B, gif_file);

查看 GIF

以上完整代码在 binglingziyu/audio-video-blog-demos 可以获取。

运行代码,生成 GIF 图片:

2.gif

代码:
audio-video-blog-demos

参考资料:

What's In A GIF

Gif 89a specification

GIF 格式解析

GIF 图片原理和储存结构

Gif 图片格式完全理解

GIF 文件格式详解

GIF 图形文件格式文档

GIF 文件格式详解

LZW 压缩算法——简明原理与实现

github.com/jefftime/lzw

https://github.com/jcraveiro

LZW compressor / decompressor

ASCII Codes Table


©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容