[Image_Codec]常见图片格式的封装及编解码-Android平台(一)BMP

[TOC]

概述

图片格式是计算机存储图片的格式,是图片在计算机中存放的样式。常见的存储的格式有很多,有不同的编码格式和封装方式,实用与各个平台和各种需求。

bmp jpg png tiff gif pcx tga exif fpx svg psd cdr pcd dxf ufo eps ai raw WMF ...

我们这里只介绍里面最常见的几种:

 bmp jpg png gif

我们采用4x2的一张图片进行介绍。这里的放大版本,4x2的图片很小,看不见,这个是放大很多倍的样图。另外,介绍时,我们默认都是采用32bit的,也就是默认的ARGB32。

图片样例

这些图片格式网络上有很多介绍的,大家也可以参考,下面我们分别来看看,我们的样例代码都是基于Android平台,但是很多都是以native的方式提供。

BMP图片格式

BMP也就是位图Bitmap,Windows标准格式图形文件,是一种与硬件设备无关的非压缩的图像文件格式,也就是采用位映射存储格式,不进行压缩,仅可以选择图像深度。所以,BMP文件所占用的空间很大,不适合网络等传输。BMP文件的图像深度可选lbit、4bit、8bit及24bit。

Bitmap文件结构

下面是Wiki上Bitmap文件的的相关信息:

属性 属性值
Filename extension .bmp, .dib
Internet media type image/bmp, image/x-bmp
Type code 'BMP', 'BMPf', 'BMPp'
Uniform Type Identifier (UTI) com.microsoft.bmp
Developed by Microsoft Corporation
Type of format Raster graphics
Open format? OSP for WMF

BMP文件存储数据时,图像的扫描方式是按从左到右、从下到上的顺序。典型的BMP图像文件由三部分组成:位图文件头数据结构,它包含BMP图像文件的类型、显示内容等信息;位图信息数据结构,它包含有BMP图像的宽、高、压缩方法,以及定义颜色等信息;像素点阵,也就是图片的具体数据。Bitap的文件结构可以参考Wiki百科的BMPfileFormat,文件格式描述如下图(此图来自wikipedia):

Bitmap文件格式

其中,Bitmap file headerDIB headerPixel array三部分是必须的。我们以一个4x2的32bit的Bitmap加以说明:

[图片上传失败...(image-f47dcf-1524794608583)]

图片非常小,前面的样图的放大后的版本,放大后的如下:

图片样例

我们用二进制将图片打开

vim -b bitmap_4x2_32bit.bmp

:%!xxd转换为16进制

00000000: 424d 5600 0000 0000 0000 3600 0000 2800  BMV.......6...(.
00000010: 0000 0400 0000 0200 0000 0100 2000 0000  ............ ...
00000020: 0000 2000 0000 0000 0000 0000 0000 0000  .. .............
00000030: 0000 0000 0000 ffff ff7f 00ff 007f 0000  ................
00000040: ff7f ff00 007f 0000 0000 0000 ff00 ff00  ................
00000050: 0000 00ff 0000                           ......

这是一张4x2 32位的bmp图片,包括Alpha通道,像素格式为ARGB32。32Bit,那么一个像素点就是32位,4字节;4x2大小,就4x2x4就是32字节。我们具体来看里面包含的信息。

  • 第一部分 BMP的头(BMP Header)
00000000: 424d 5600 0000 0000 0000 3600 0000 
偏移量Offset 大小Size(字节) 十六进制值Hex Value 值Value 描述Description
0h 2 424d BM BMP的ID(42,4d)
2h 4 5600 0000 0x56字节(头54+像素32=86字节) BMP文件的大小
6h 2 0000 没有用到 应用指定的
8h 2 0000 没有用到 应用指定的
Ah 4 3600 0000 0x36字节(14+40) Bitmap数据的起始位置

BMP的头大小为14字节。而下一部分要说道的DIB Header为40字节,所以像素的位置为0x36字节(14+40)。

  • 第二部分 DIB Header
00000000: ---- ---- ---- ---- ---- ---- ---- 2800  BMV.......6...(.
00000010: 0000 0400 0000 0200 0000 0100 2000 0000  ............ ...
00000020: 0000 2000 0000 0000 0000 0000 0000 0000  .. .............
00000030: 0000 0000 0000
偏移量Offset 大小Size(字节) 十六进制值Hex Value 值Value 描述Description
Eh 4 2800 0000 0x28字节 DIB Header的大小
12h 4 0400 0000 4 pixels Bitmap的宽,宽高是用是像素表的
16h 4 0200 0000 2 pixels Bitmap的高
1Ah 2 0100 1 plane 会用到是颜色板数量
1Ch 2 2000 32 bits 每个像素点占的bit位
1Eh 4 0000 0000 0 BI_RGB, 没有像素被压缩
22h 4 2000 0000 32字节 Bitmap raw数据的大小,包括对齐的部分
26h 4 0000 0000 0 pixels/metre horizontal 打印信息没有指定
2Ah 4 0000 0000 0 pixels/metre vertical 打印信息没有指定
2Eh 4 0000 0000 0 colors 调色板数量
32h 4 0000 0000 0 important colors 0表示所有color都重要
  • 第三部分 像素点数据 Pixels
00000030: ---- ---- ---- ffff ff7f 00ff 007f 0000  ................
00000040: ff7f ff00 007f 0000 0000 0000 ff00 ff00  ................
00000050: 0000 00ff 0000                           ......
偏移量Offset 大小Size(字节) 十六进制值Hex Value 值Value 描述Description
36h 4 ffff ff7f 255 255 255 127 白色Alpha为7f,ARGB32,采用小端模式
3Ah 4 00ff 007f 0 255 0 127 绿色Alpha为7f,Bitmap扫描采用从做到右,下到上
3Eh 4 0000 ff7f 0 0 255 127 红色Alpha为7f,从左到右BGRA
42h 4 ff00 007f 255 0 0 127 蓝色Alpha为7f
46h 4 0000 0000 0 0 0 0 黑色Alpha为0
4Ah 4 0000 ff00 0 0 255 0 红色Alpha为0
4Eh 4 ff00 0000 255 0 0 0 蓝色Alpha为0
52h 4 00ff 0000 0 255 0 0 绿色Alpha为0

以上就是BMP的文件封装的简单介绍。

试想一下,如果是一张2x2 24bit的BMP图片会是什么样的呢?参照前面的描述。需要变的有一下几个地方:
每像素的位数:24, 0x18;3字节
尺寸:2x2 0x02
Pixel的大小:2x2x3 12字节,0x0c
文件的大小:54 + 12 66字,0x42
Pixels的值,就黑白,白黑的矩阵吧:

ffff ff 0000 00
000 00 ffff ff

修改文件吧~

00000000: 424d 4200 0000 0000 0000 3600 0000 2800  BMV.......6...(.
00000010: 0000 0200 0000 0200 0000 0100 1800 0000  ............ ...
00000020: 0000 0c00 0000 0000 0000 0000 0000 0000  .. .............
00000030: 0000 0000 0000 ffff ff00 0000 0000 00ff  ................
00000040: ffff                                     ................

:%!xxd -r转成二进制,:wq保存退出。打开我们编辑后的图片,看看是什么样的?

打不开吧?Why?

这是因为,宽需要做一个8字节的对齐,我们来看对齐后,应该是什么样的。

原来是2x3,6字节,对齐后应该是8字节。pixel的大小是不变为16字节了,0x10;文件大小为0x46,像素矩阵为:

ffff ff00 0000 0000
000 00ff ffff 0000

后面的0000的补齐的,不是bitmap的数据,修改文件吧~

再打开bitmap文件试试,可以打开吧!只是2x2的图片实在太小了,很难看见,下面是放大后的效果:

2x2 RGB24的BMP图片

Bitmap文件 解析

Bitmap中的像素点数据Pixels是没有经过压缩的,我们读出来后就是图片的数据值,所以没有解压一说。但是需要我们从一个Bitmap文件中,将需要的信息获取出来。我们勉强也把这个过程称为解压吧!

样例代码请参考github Codec-BitmapCodec

我们的样例的代码是用Java实现的,很简单,只包括我们前面介绍的Bitmap文件必现的信息。

public class BitmapFile {
    String fileName;

    byte type[];
    byte data[];

    byte pixels[];
    byte outPixels[];

    BitmapFileHeader fileHeader;

    BitmapDibHeader dibHeader;

定义BitmapFile类来描述一个Bitmap文件,type是类型,Bitmap的type为42 4d;data Bitmap读出来的所有数据。文件头,用类BitmapFileHeader进行描述;Dib头用BitmapDibHeader进行描述。pixels 从Bitmap文件中读出来的像素点,outPixels是准备写到Bitmap文件中的像素点。

BitmapFileHeader定义如下:

public class BitmapFileHeader {
    int     fileSize;
    short   reserved1;
    short   reserved2;
    int     pixelsOffset;
}

BitmapDibHeader定义如下:

public class BitmapDibHeader {
    int     dibSize;
    int     width;
    int     height;
    short   planes;
    short   bitPerPixel;
    int     compressed;
    int     pixelSize;
    int     xPixelsPerMeter;
    int     yPixelsPerMeter;
    int     clrUsed;
    int     clrImportant;
    int     bytePerPixel;
}

bytePerPixel是加的,每个像素占多少个字节。

解析Bitmap文件的时候,直接读取文件中的数据,将文件中的数据按照封装读出来。

    public boolean readBitmap() throws Exception {
        DataReader reader = new DataReader();
        reader.readFromFile(fileName);

        type = reader.readType();

        fileHeader.fileSize = reader.readInt();
        fileHeader.reserved1 = reader.readShort();
        fileHeader.reserved2 = reader.readShort();
        fileHeader.pixelsOffset = reader.readInt();

        dibHeader.dibSize = reader.readInt();
        dibHeader.width = reader.readInt();
        dibHeader.height = reader.readInt();
        dibHeader.planes = reader.readShort();
        dibHeader.bitPerPixel = reader.readShort();
        dibHeader.compressed = reader.readInt();
        dibHeader.pixelSize = reader.readInt();
        dibHeader.xPixelsPerMeter = reader.readInt();
        dibHeader.yPixelsPerMeter = reader.readInt();
        dibHeader.clrUsed = reader.readInt();
        dibHeader.clrImportant = reader.readInt();

        data = reader.readBytesFrom(0);

        pixels = reader.readBytesFrom(fileHeader.pixelsOffset);

        return true;
    }

将我们前面自己编辑的bitmap_2x2_24bit.bmp文件读出来看看:

    Bitmap Info 
    type:424d
    fileSize:0x46
    reserved1:0x0
    reserved2:0x0
    pixelsOffset:0x36
    dibSize:0x28
    width:0x2
    height:0x2
    planes:0x1
    bitPerPixel:0x18
    compressed:0x0
    pixelSize:0x10
    xPixelsPerMeter:0x0
    yPixelsPerMeter:0x0
    clrUsed:0x0
    clrImportant:0x0
    pixels:ffffff0000000000000000ffffff0000

我们解析出来的信息是对的。

Bitmap文件封装

我们将前面的bitmap_2x2_24bit.bmp扩展为32bit,也就是加上Alpha通道;放大100倍(w,h都放大10倍)后,再保存为Bitmap文件。

    private void buildBitmapAndSave() {
        BitmapFile bmpFile  = new BitmapFile();

        BitmapDibHeader dibHeader = new BitmapDibHeader();
        dibHeader.dibSize = 0x28;
        dibHeader.width = 20;
        dibHeader.height = 20;
        dibHeader.planes = 1;
        dibHeader.bitPerPixel = 32;
        dibHeader.bytePerPixel = 4;
        dibHeader.pixelSize = dibHeader.width * dibHeader.height * dibHeader.bytePerPixel;

        bmpFile.setDibHeader(dibHeader);

        BitmapFileHeader fileHeader = new BitmapFileHeader();
        fileHeader.fileSize = 0x36 + dibHeader.pixelSize;
        fileHeader.pixelsOffset = 0x36;
        fileHeader.reserved1 = Utils.char2short(new char[]{'T', 'S'});
        fileHeader.reserved2 = Utils.char2short(new char[]{'C', 'J'});

        bmpFile.setFileHeader(fileHeader);

        // raw data 20x20x4,already 8bit align.
        byte[] pixels = new byte[]{
             ... ...
        };

        bmpFile.setOutPixels(pixels);

        try {
            String fileName = "/sdcard/bitmap_20_20_32bit.bmp";
            int size = bmpFile.saveBitmap(fileName);
            MyLog.d("buildBitmapAndSave " + size + " bytes to " + fileName);
        } catch (Exception e) {
            MyLog.e("buildBitmapAndSave fail " + e.toString());
        }
    }

一个像素点4个byte,pixels总共20x20x4,1600个byte。量比较大,我们就不列出来了。红色我们用的0000 ff00,ff是Alpha值,绿色就是00ff 0000,不透明。直接给个图吧 bitmap_20x20_32bit.bmp

20x20 ARGB32的Bitmap图片(bmp上传不了,转为了jpg)

Bitmap接介绍到这里,实例代码可以高github里面去下载哦~

Codec-BitmapCodec

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容