在学习了AudioTrack播放pcm数据之后,又了解到很多APP不是使用MediaPlayer对音视频文件进行播放的。而是使用解码器,从音视频文件中解码出pcm原生数据,然后使用AudioTrack.java将音频播放出来。所有对其实现过程产生了兴趣,并进行了学习。总结如下:
1. MediaExtractor和MediaCodec的初认知:
MediaExtractor:a. 将音视频文件解析出音轨和视轨数据; b.可以获取音/视轨的参数信息(如getTrackFormat()获得mediaFormat后,从mediaFormat中可得到视频的width/height/duration等数据)
MediaCodec:将音视频文件解码成可以用Surface显示和用AudioTrack播放类型的数据。
2.MediaExtractor/MediaCodec基本使用方法:
A. 视频数据的显示
MediaExtractor videoExtractor = new MediaExtractor(); //创建对象 videoExtractor.setDataSource(sourcePath); //设置需播放的视频文件路径 ------------------------------------------------------------------------------- //接下来需要做的是,从videoExtractor中找到视轨的id,方法如下: int videoTrackIndex = -1; //定义trackIndex为视轨的id for (int i = 0; i < videoExtractor.getTrackCount(); i++) { //在videoExtractor的所以Track中遍历,找到视轨的id MediaFormat mediaFormat = videoExtractor.getTrackFormat(i); //获得第id个Track对应的MediaForamt String mime = mediaFormat.getString(MediaFormat.KEY_MIME); //再获取该Track对应的KEY_MIME字段 if (mime.startsWith("video/")) { //视轨的KEY_MIME是以"video/"开头的,音轨是"audio/" trackIndex = i; break; } } ------------------------------------------------------------------------------- videoExtractor.selectTrack(videoTrackIndex); //选择视轨所在的轨道子集(这样在之后调用readSampleData()/getSampleTrackIndex()方法时候,返回的就只是视轨的数据了,其他轨的数据不会被返回) MediaFormat mediaFormat = videoExtractor.getTrackFormat(videoTrackIndex); //根据视轨id获得对应的MediaForamt int width = mediaFormat.getInteger(MediaFormat.KEY_WIDTH); int height = mediaFormat.getInteger(MediaFormat.KEY_HEIGHT); int time = mediaFormat.getInteger(MediaFormat.KEY_DURATION); //从得到的MediaFormat中,可以获取视频的相关信息,视频的长/宽/时长等 -------------------------------------------------------------------------------- 在通过MediaExtractor获得需要解码的音轨的id后,就可以创建对应的MediaCodec来解析数据了 mVideoCodec = MediaCodec.createDecoderByType(mediaFormat.getString(MediaFormat.KEY_MIME)); //参数为MediaFormat类中的MIMETYPE mVideoCodec.configure(mediaFormat, surface, null, 0); //第一个参数是待解码的数据格式(也可用于编码操作);第二个参数是设置surface,用来在其上绘制解码器解码出的数据;第三个参数于数据加密有关;第四个参数上1表示编码器,0是否表示解码器呢?? mVideoCodec.start(); //当configure好后,就可以调用start()方法来请求向MediaCodec的inputBuffer中写入数据了 ------------------------------------------------------------------------------- 知识点:MediaCodec类中有三个方法与数据读写有关:queueInputBuffer()/dequeueInputBuffer()/dequeueOutputBuffer() MediaCodec中有维护这两个缓冲区,分别存放的是向MediaCodec中写入的数据,和经MediaCodec解码后写出的数据 dequeueInputBuffer(): Returns the index of an input buffer to be filled with valid data dequeueOutputBuffer():Returns the index of an output buffer that has been successfully decoded queueInputBuffer(): After filling a range of the input buffer at the specified index submit it to the component 接下来要做到就是,向MediaCodec的inputBuffer中写入数据,而数据就是来自上面MediaExtractor中解析出的Track,代码如下: ByteBuffer[] inputBuffers = mVideoCodec.getInputBuffers(); //获取MediaCodec中等待数据写入的ByteBuffer的集合,大概有10个ByteBuffer 上面这个方法获取的是整个待写入数据的ByteBuffer的集合,在MediaExtractor向MediaCodec中写入数据的过程中,需要判断哪些ByteBuffer 是可用的,这就可以通过dequeueInputBuffer得到。 int inputBufferIndex = mediaCodec.dequeueInputBuffer(TIMEOUT_US); //得到那个可以使用的ByteBuffer的id if (inputBufferIndex >= 0) { //返回的inputBufferIndex为-1,说明暂无可用的ByteBuffer ByteBuffer inputBuffer = inputBuffers[inputBufferIndex]; //有可以就从inputBuffers中拿出那个可用的ByteBuffer的对象 int sampleSize = mediaExtractor.readSampleData(inputBuffer, 0); //把MediaExtractor中的数据写入到这个可用的ByteBuffer对象中去,返回值为-1表示MediaExtractor中数据已全部读完 if (sampleSize < 0) { isMediaEOS = true; mediaCodec.queueInputBuffer(inputBufferIndex, 0, 0, 0, MediaCodec.BUFFER_FLAG_END_OF_STREAM); } else { mediaCodec.queueInputBuffer(inputBufferIndex, 0, sampleSize, mediaExtractor.getSampleTime(), 0); //将已写入数据的id为inputBufferIndex的ByteBuffer提交给MediaCodec进行解码 mediaExtractor.advance(); //在MediaExtractor执行完一次readSampleData方法后,需要调用advance()去跳到下一个sample,然后再次读取数据 } } 最后一步的作用容易被忽视掉,但是如果没有的话,也会导致视频无法播放出来,代码如下: int outputBufferIndex = mVideoCodec.dequeueOutputBuffer(videoBufferInfo, TIMEOUT_US); //获得已经成功解码的ByteBuffer的id switch (outputBufferIndex) { case MediaCodec.INFO_OUTPUT_FORMAT_CHANGED: Log.v(TAG, "format changed"); break; case MediaCodec.INFO_TRY_AGAIN_LATER: Log.v(TAG, "超时"); break; case MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED: Log.v(TAG, "output buffers changed"); break; default: mVideoCodec.releaseOutputBuffer(outputBufferIndex, true); break; //将该ByteBuffer释放掉,以供缓冲区的循环使用。如果没有这一步的话,会导致上面返回的inputBufferIndex一直为-1,使数据读写操作无法进行下去。如果在configure中配置了surface,则首先将缓冲区中数据发送给surface,surface一旦不再使用,就将缓冲区释放给MediaCodec 注意:本段的数据读写操作,应该是循环的。它的中断/结束条件是:a.停止播放视频 b.视频播放结束(结束的标志可从sampleSize是否为-1来判断)
B. 音频数据的播放
音频数据的播放和视频数据的显示,大体流程相同,不过有几点需要说明下:
MediaExtractor audioExtractor = new MediaExtractor(); audioExtractor.setDataSource(sourcePath); for (int i = 0; i < audioExtractor.getTrackCount(); i++) { MediaFormat mediaFormat = audioExtractor.getTrackFormat(i); String mime = mediaFormat.getString(MediaFormat.KEY_MIME); if (mime.startsWith("audio/")) { audioExtractor.selectTrack(i); int audioChannels = mediaFormat.getInteger(MediaFormat.KEY_CHANNEL_COUNT); int audioSampleRate = mediaFormat.getInteger(MediaFormat.KEY_SAMPLE_RATE); int minBufferSize = AudioTrack.getMinBufferSize(audioSampleRate,(audioChannels == 1 ? AudioFormat.CHANNEL_OUT_MONO : AudioFormat.CHANNEL_OUT_STEREO), AudioFormat.ENCODING_PCM_16BIT); int maxInputSize = mediaFormat.getInteger(MediaFormat.KEY_MAX_INPUT_SIZE); audioInputBufferSize = minBufferSize > 0 ? minBufferSize * 4 : maxInputSize; audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC,audioSampleRate,(audioChannels == 1 ? AudioFormat.CHANNEL_OUT_MONO : AudioFormat.CHANNEL_OUT_STEREO), AudioFormat.ENCODING_PCM_16BIT,audioInputBufferSize,AudioTrack.MODE_STREAM); audioTrack.play(); audioCodec = MediaCodec.createDecoderByType(mime); audioCodec.configure(mediaFormat, null, null, 0); //因为是音轨,所以第二参数为null audioCodec.start(); ByteBuffer[] outputBuffers = audioCodec.getOutputBuffers(); ByteBuffer[] inputBuffers = audioCodec.getInputBuffers(); MediaCodec.BufferInfo audioBufferInfo = new MediaCodec.BufferInfo();
然后就是将MediaExtractor的数据写入到MediaCodec中,然后利用dequeueOutputBuffer()将已经成功解析出pcm数据的ByteBuffer的id得到,获得pcm保存的数组outputBuffer。
int outputBufferIndex = audioCodec.dequeueOutputBuffer(audioBufferInfo, TIMEOUT_US); switch (outputBufferIndex) { case MediaCodec.INFO_OUTPUT_FORMAT_CHANGED: Log.v(TAG, "format changed"); break; case MediaCodec.INFO_TRY_AGAIN_LATER: Log.v(TAG, "超时"); break; case MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED: outputBuffers = audioCodec.getOutputBuffers(); //注意:当dequeueOutputBuffer返回的id为-3时,说明缓存区发生了改变,必须重新通过getOutputBuffers来获得新的outputBuffer对象,不能再使用之前创建的了。 Log.v(TAG, "output buffers changed"); break; default: ByteBuffer outputBuffer = outputBuffers[outputBufferIndex]; //1. 视频可以直接显示在Surface上,音频需要获取pcm所在的ByteBuffer byte[] tempBuffer = new byte[outputBuffer.limit()] outputBuffer.position(0); outputBuffer.get(tempBuffer, 0, outputBuffer.limit()); //2.将保存在ByteBuffer的数据,转到临时的tempBuffer字节数组中去 outputBuffer.clear(); if (audioTrack != null) audioTrack.write(tempBuffer, 0, audioBufferInfo.size); //3.最后写到AudioTrack中 audioCodec.releaseOutputBuffer(outputBufferIndex, false); //4. 释放该id的ByteBuffer break; }