音视频学习系列第(六)篇---音视频的分离与合成

音视频系列

什么是音视频的分离和合成

分离就是将视频1的声音和图像分别取出来
合成就是将视频1的图像和非视频1的声音组合成一个新的视频

如何进行音视频的分离和合成

安卓提供了两个API来帮助我们完成这个操作
MediaExtractor用于分离视频
MediaMuxer用于合成视频

下面我就来介绍一下这两个API的使用

MediaExtractor

分离音频
1.设置音频源

 MediaExtractor  audioExtractor = new MediaExtractor();
 audioExtractor.setDataSource(audioPath);

2.获取源文件中轨道的数量,并遍历找到我们需要的音频轨

  for (int i = 0; i < audioExtractor.getTrackCount(); i++) {
      MediaFormat format = audioExtractor.getTrackFormat(i);
      if (format.getString(MediaFormat.KEY_MIME).startsWith("audio/")) {
          srcATrackIndex = i;
          audioTrackIndex = muxer.addTrack(format);
          break;
      }
  }

分离视频
1.设置视频源

MediaExtractor  videoExtractor = new MediaExtractor();
videoExtractor.setDataSource(videoPath);

2.获取视频源文件中的轨道数,并遍历找到我们所需要的视频轨

for (int i = 0; i < videoExtractor.getTrackCount(); i++) {
       MediaFormat format = videoExtractor.getTrackFormat(i);
       if (format.getString(MediaFormat.KEY_MIME).startsWith("video/")) {
            srcVTrackIndex = i;
            videoTrackIndex = muxer.addTrack(format);
            break;
        }
 }

音频和视频额的分离方法一模一样,区别在于MediaFormat类型的不同,MediaFormat封装了媒体的数据格式信息

MediaMuxer

如何合成视频?
1.设置合成后视频的路径和格式

MediaMuxer  muxer = new MediaMuxer(outPath, MediaMuxer.OutputFormat.MUXER_OUTPUT_MPEG_4);

2.将MediaExtractor分离出来的音轨和视轨添加到自己的轨道中

audioTrackIndex = muxer.addTrack(format);
videoTrackIndex = muxer.addTrack(format);

3.添加完所有轨道后start

muxer.start();

4.采集音频源的音轨样本

audioExtractor.selectTrack(srcATrackIndex);   //移动到音频轨上
        if (audioTrackIndex != -1) {
            MediaCodec.BufferInfo info = new MediaCodec.BufferInfo();
            info.presentationTimeUs = 0;
            ByteBuffer buffer = ByteBuffer.allocate(100 * 1024);
            while (true) {
                int sampleSize = audioExtractor.readSampleData(buffer, 0);
                if (sampleSize < 0) {
                    //没有可获取的样本,退出循环
                    break;
                }

                info.offset = 0;
                info.size = sampleSize;
                info.flags = MediaCodec.BUFFER_FLAG_SYNC_FRAME;
                info.presentationTimeUs = audioExtractor.getSampleTime();

                muxer.writeSampleData(audioTrackIndex, buffer, info);//将样本写入新的轨道
                audioExtractor.advance(); //进入下一个样本
            }
 }

5.采集视频源的视频轨样本

videoExtractor.selectTrack(srcVTrackIndex);   //移动到视频轨上
        if (videoTrackIndex != -1) {
            MediaCodec.BufferInfo info = new MediaCodec.BufferInfo();
            info.presentationTimeUs = 0;
            ByteBuffer buffer = ByteBuffer.allocate(100 * 1024);
            while (true) {
                int sampleSize = videoExtractor.readSampleData(buffer, 0);
                if (sampleSize < 0) {
                    //没有可获取的样本,退出循环
                    break;
                }

                info.offset = 0;
                info.size = sampleSize;
                info.flags = MediaCodec.BUFFER_FLAG_SYNC_FRAME;
                info.presentationTimeUs = videoExtractor.getSampleTime();

                muxer.writeSampleData(videoTrackIndex, buffer, info);//将样本写入新的轨道
                videoExtractor.advance(); //进入下一个样本
            }
 }

6.停止并释放资源

muxer.stop();
muxer.release();

完整代码

public class MediaUtil {

private static final String TAG = "MediaUtil";
/**
 * @param audioPath 音频文件路劲
 * @param videoPath 视频文件路径
 * @param outPath   合成之后的保存路径
 */
public static void combineVideo(String audioPath, String videoPath, String outPath) {
    MediaMuxer muxer = null;
    MediaExtractor audioExtractor = null;
    MediaExtractor videoExtractor = null;
    try {
        muxer = new MediaMuxer(outPath, MediaMuxer.OutputFormat.MUXER_OUTPUT_MPEG_4);

        //找到音频文件的音频轨
        audioExtractor = new MediaExtractor();
        audioExtractor.setDataSource(audioPath);
        int srcATrackIndex = -1;    //音频源的音频轨
        int audioTrackIndex = -1; //音频轨添加到muxer后返回的新的轨道
        //在此循环,目的是找到我们需要的音频轨
        for (int i = 0; i < audioExtractor.getTrackCount(); i++) {
            MediaFormat format = audioExtractor.getTrackFormat(i);
            if (format.getString(MediaFormat.KEY_MIME).startsWith("audio/")) {
                srcATrackIndex = i;
                audioTrackIndex = muxer.addTrack(format);
                break;
            }
        }

        Log.d(TAG,"音频轨源索引:"+srcATrackIndex+" 音频轨新索引:"+audioTrackIndex);

        //找到视频文件的视频轨
        videoExtractor = new MediaExtractor();
        videoExtractor.setDataSource(videoPath);
        int srcVTrackIndex = -1;  //视频源的视频轨
        int videoTrackIndex = -1;  //视频轨添加到muxer后返回的新的轨道
        for (int i = 0; i < videoExtractor.getTrackCount(); i++) {
            MediaFormat format = videoExtractor.getTrackFormat(i);
            if (format.getString(MediaFormat.KEY_MIME).startsWith("video/")) {
                srcVTrackIndex = i;
                videoTrackIndex = muxer.addTrack(format);
                break;
            }
        }

        Log.d(TAG,"视频轨源索引:"+srcVTrackIndex+" 视频频轨新索引:"+videoTrackIndex);


        //添加完所有轨道后start
        muxer.start();
        Log.d(TAG,"开始合成视频...");

        //封装音频track
        audioExtractor.selectTrack(srcATrackIndex);   //移动到音频轨上
        if (audioTrackIndex != -1) {
            MediaCodec.BufferInfo info = new MediaCodec.BufferInfo();
            info.presentationTimeUs = 0;
            ByteBuffer buffer = ByteBuffer.allocate(100 * 1024);
            while (true) {
                int sampleSize = audioExtractor.readSampleData(buffer, 0);
                if (sampleSize < 0) {
                    //没有可获取的样本,退出循环
                    break;
                }

                info.offset = 0;
                info.size = sampleSize;
                info.flags = MediaCodec.BUFFER_FLAG_SYNC_FRAME;
                info.presentationTimeUs = audioExtractor.getSampleTime();

                muxer.writeSampleData(audioTrackIndex, buffer, info);//将样本写入新的轨道
                audioExtractor.advance(); //进入下一个样本
            }
        }
        Log.d(TAG,"音频轨样本采集完成");

        //封装视频track
        videoExtractor.selectTrack(srcVTrackIndex);   //移动到视频轨上
        if (videoTrackIndex != -1) {
            MediaCodec.BufferInfo info = new MediaCodec.BufferInfo();
            info.presentationTimeUs = 0;
            ByteBuffer buffer = ByteBuffer.allocate(100 * 1024);
            while (true) {
                int sampleSize = videoExtractor.readSampleData(buffer, 0);
                if (sampleSize < 0) {
                    //没有可获取的样本,退出循环
                    break;
                }

                info.offset = 0;
                info.size = sampleSize;
                info.flags = MediaCodec.BUFFER_FLAG_SYNC_FRAME;
                info.presentationTimeUs = videoExtractor.getSampleTime();

                muxer.writeSampleData(videoTrackIndex, buffer, info);//将样本写入新的轨道
                videoExtractor.advance(); //进入下一个样本
            }
        }
        muxer.stop();
        Log.d(TAG,"视频轨样本采集完成");
        Log.d(TAG,"视频合成完毕:"+outPath);
    } catch (IOException e) {
        e.printStackTrace();
        Log.d(TAG,"合成出错:"+e.getMessage());
    } finally {
        //释放资源
        if(audioExtractor!=null){
            audioExtractor.release();
        }
        if(videoExtractor!=null){
            videoExtractor.release();
        }
        if(muxer!=null){
            muxer.release();
        }
    }
}
}

踩坑总结

问题1
MPEG4Writer: Unsupported mime 'audio/mpeg'
当音频文件是mp3时会报这个错误 ,需要的格式是AAC,m4a
stackoverflow
我将两个源文件都替换了mp4格式后不再报错了

问题2
WVMExtractor: Failed to open libwvm.so: dlopen failed: library "libwvm.so" not found
小米5 6.0.1机型上出现的问题,意思是缺少so库

由于我就是这款手机,也没测过其他手机有木有这个so库
后续我会用其他手机在测试这个问题

代码地址

MediaUtil类的封装放置在libplayer下的util包下
方法调用放置在app/demo/media/track下

你可能感兴趣的:(音视频学习系列第(六)篇---音视频的分离与合成)