详解Android开发之MP4文件转GIF文件
一基本实现原理
在介绍具体实现过程之前,先简单说下基本原理和实现步骤,在解决相对比较复杂的问题,我习惯先理清主要原理步骤,不要一开始就被繁琐细节绊住,待具体实现时再逐个攻破。下面是主要步骤:
1、视频文件的读取:包括录制和本地文件读取
2、将需要转换的视频部分解析为Bitmap序列
3、将解析好的Bitmap序列编码生成GIF文件
二视频文件的读取
视频文件的读取比较简单,没什么特别需要说的地方,这里简单贴出视频读取的核心部分代码,详细实现可以Google一下就行了。
privateView.OnClickListenerclickListener=newView.OnClickListener(){ @Override publicvoidonClick(Viewv){ Intentintent=newIntent(); intent.setType("video/*"); intent.setAction(Intent.ACTION_GET_CONTENT); startActivityForResult(Intent.createChooser(intent,"SelectVideo"),SELECT_VIDEO); } }; @Override protectedvoidonActivityResult(intrequestCode,intresultCode,Intentdata){ if(requestCode==REQUEST_SELECT_VIDEO){ if(resultCode==RESULT_OK){ UrivideoUri=data.getData(); filePath=getRealFilePath(videoUri); } } }
三视频文件的解析
视频文件读取成功后,接下来要做的就是解析视频文件,选取需要转换的视频片段,提取Bitmap序列。下面来看下具体实现,提取Bitmap序列就是根据给定的起始时间和结束时间以及帧率从视频文件中获取相应的Bitmap,本文主要是利用MediaMetadataRetriever提供的API来实现的,在看代码前可以先看下MediaMetadataRetriever的API文档,该类的核心功能就是获取视频的帧和元数据,下面是核心实现代码:
publicList<Bitmap>createBitmaps(Stringpath){ MediaMetadataRetrievermmr=newMediaMetadataRetriever(); mmr.setDataSource(path); doubleinc=1000*1000/fps; for(doublei=begin;i<end;i+=inc){ Bitmapframe=mmr.getFrameAtTime((long)i,MediaMetadataRetriever.OPTION_CLOSEST); if(frame!=null){ bitmaps.add(scale(frame)); } } returnbitmaps; } privateBitmapscale(Bitmapbitmap){ returnBitmap.createScaledBitmap(bitmap, width>0?width:bitmap.getWidth(), height>0?height:bitmap.getHeight(), true); }
四生成GIF文件
拿到要生成GIF的Bitmap序列,接下来需要做的就是将Bitmap序列中的数据按照GIF的文件格式编码,生成最终的GIF文件。目标很明确,接下来就看具体实现过程了。
1.GIF格式简介
生成GIF文件之前有必要介绍下GIF的存储格式,GIF格式的相关文章比较多,这里也没必要太详细的介绍,只是简单说下后面程序中会用到的方面。
GIF图象是基于颜色列表的(存储的数据是该点的颜色对应于颜色列表的索引值),最多只支持8位(256色)。GIF文件内部分成许多存储块,用来存储多幅图象或者是决定图象表现行为的控制块,用以实现动画和交互式应用。GIF文件还通过LZW压缩算法压缩图象数据来减少图象尺寸。
GIF文件内部是按块划分的,包括控制块和数据块两种。控制块是控制数据块行为的,根据不同的控制块包含一些不同的控制参数;数据块只包含一些8-bit的字符流,由它前面的控制块来决定它的功能,每个数据块0到255个字节,数据块的第一个字节指出这个数据块大小(字节数),计算数据块的大小时不包括这个字节,所以一个空的数据块有一个字节,那就是数据块的大小0x00。
2.GIF文件写入
刚开始接触GIF文件会觉得比较复杂,存储格式、编码格式等都比Bitmap要复杂的多,但其实可以把问题简单化理解,生成GIF和生成Bitmap原理类似,就是按照规定的格式写文件就行了,不用太纠结内部细节,否则就会陷入繁琐的细节(俗称钻牛角尖)而忽略了最终目的只是为了生成GIF文件。下面就来看下有哪些文件部分需要写入的:
提取Bitmap的像素值
首先需要将上面得到的Bitmap的像素值提取出来,方便后面把像素值写入到GIF文件中,在提取像素值的同时,生成GIF文件所需要的颜色表,生成颜色表过程比较复杂,这里就不贴出源码,感兴趣的可以Google一下颜色量化算法,不感兴趣的直接用现成的就好,下面是提取像素值的具体实现:
protectedvoidgetImagePixels(){ intw=image.getWidth(); inth=image.getHeight(); pixels=newbyte[w*h*3]; for(inti=0;i<h;i++){ intstride=w*3*i; for(intj=0;j<w;j++){ intp=image.getPixel(j,i); intstep=j*3; intoffset=stride+step; //blue pixels[offset+0]=(byte)((p&0x0000FF)>>0); //green pixels[offset+1]=(byte)((p&0x00FF00)>>8); //red pixels[offset+2]=(byte)((p&0xFF0000)>>16); } } }
GIF文件头(Header)
文件头部分总共6个字节,包括:GIF署名和版本号,GIF署名由3个字符"GIF"组成,共3个字节,版本号也是由3个字节组成,可以为"87a"或"89a"(分别为1987年和1989年版本),实现代码如下:
//写入文件头 protectedvoidwriteHeader()throwsIOException{ writeString("GIF89a"); } protectedvoidwriteString(Strings)throwsIOException{ for(inti=0;i<s.length();i++){ out.write((byte)s.charAt(i)); } }
逻辑屏幕标识符(LogicalScreenDescriptor)
文件头的后面是逻辑屏幕标识符(LogicalScreenDescriptor),这一部分由7个字节组成,定义了GIF图象的大小、颜色深度、背景色以及有无全局颜色列表和颜色列表的索引数。实现代码如下:
//写入逻辑屏幕标识符 protectedvoidwriteLSD()throwsIOException{ writeShort(width);//写入图像宽度 writeShort(height);//写入图像高度 out.write((0x80|//全局颜色列表标志置1 0x70|//确定图象的颜色深度(7+1=8) 0x00|//全局颜色列表分类排列置为0 0x07));//颜色列表的索引数(2的7+1次方) out.write(0);//背景颜色(在全局颜色列表中的索引) out.write(0);//像素宽高比默认1:1 } protectedvoidwriteShort(intvalue)throwsIOException{ out.write(value&0xff); out.write((value>>8)&0xff); }
逻辑屏幕标识符部分结构稍微复杂些,如果不知道每一位代表什么意思可以参考:GIF图形文件格式文档中的逻辑屏幕标识符部分。
全局颜色列表(GlobalColorTable)
全局颜色列表必须紧跟在逻辑屏幕标识符后面,每个颜色列表索引条目由三个字节组成,按R、G、B的顺序排列,具体生成颜色表的实现可以看源码部分,由于生成过程比较复杂,这里就不贴颜色表生成的代码了,下面是写入颜色表的代码:
//写入颜色表 protectedvoidwritePalette()throwsIOException{ out.write(colorTab,0,colorTab.length); intn=(3*256)-colorTab.length; for(inti=0;i<n;i++){ out.write(0); } }
图形控制扩展(GraphicControlExtension)
这一部分是可选的,89a版本才支持,可以放在一个图象块(包括图象标识符、局部颜色列表和图象数据)或文本扩展块的前面,用来控制跟在它后面的第一个图象(或文本)的渲染(Render)形式,下面实现代码:
protectedvoidwriteGraphicCtrlExt()throwsIOException{ out.write(0x21);//扩展块标识,固定值0x21 out.write(0xf9);//图形控制扩展标签,固定值0xf9 out.write(4);//块大小,固定值4 out.write(0|//1:3保留位 0|//4:6不使用处置方法 0|//7用户输入标志置0 0);//8透明色标志置0 writeShort(delay);//延迟时间 out.write(0);//透明色索引值 out.write(0);//块终结器,固定值0 }
图象标识符(ImageDescriptor)
一个GIF文件内可以包含多幅图象,一幅图象结束之后紧接着下是一幅图象的标识符,图象标识符以0x2C(',')字符开始,定义紧接着它的图象的性质,包括图象相对于逻辑屏幕边界的偏移量、图象大小以及有无局部颜色列表和颜色列表大小,由10个字节组成,下面是实现代码:
protectedvoidwriteImageDesc()throwsIOException{ out.write(0x2c);//图象标识符开始,固定值为0x2c writeShort(0);//x方向偏移 writeShort(0);//y方向偏移 writeShort(width);//图像宽度 writeShort(height);//图像高度 out.write(( 0x80|//局部颜色列表标志置1 0x00| 0x00| 0x07));//局部颜色列表的索引数(2的7+1次方) }
图象数据(ImageData)
GIF图象数据使用了LZW压缩算法,大大减小了图象数据的大小,具体的LZW压缩算法可以Google一下,程序实现部分可以参考文章底部的源码链接。下面是图像数据的写入实现:
protectedvoidwritePixels()throwsIOException{ LZWEncoderencoder=newLZWEncoder( width,height,indexedPixels,colorDepth); encoder.encode(out); }
文件终结器(Trailer)
这一部分只有一个字节,标识一个GIF文件结束,固定值为0x3B,实现代码:
publicvoidfinish()throwsIOException{ out.write(0x3b); out.flush(); out.close(); }
总结
到目前为止,将MP4文件转换为GIF文件的实现过程基本完成,如果需要对GIF文件进行裁剪、添加水印等处理的话,可以在Bitmap序列写入GIF之前,对Bitmap进行相应的处理即可,如果有什么问题欢迎交流学习。希望本文的内容对大家的学习工作能有所帮助。