音视频分离与合成

//将视频分离成无声视频和纯音频,再合成为原来的视频
* AudioRecord 录制音频bugger,可以直接将buffer保存为pcm的文件,手动加音频头变成wav的无损音频;也可以直接给muxer自动加头生成音频,也可以将bugger给mediacodec进行编码亚索,压缩后通过muxer生成体积小的音频
* Mediacodec  实现编解码,及实现buffer的压缩和解压
* Muxer       将buffer 进行混合封装,生成音频和视频,音频封装时自动加音频头;可以将音频buffer和视频buffer混合成视频,可以将音频buffer混合成混合成音频文件,可以将视频buffer混合成视频文件
* MediaExtractor 分离获取器,通过此类可以获取文件的音轨、视轨、字母轨等各种轨道和信息并获取buffer数据,目的就是buffer数据,有了buffer数据就可进行编码解码压缩混合成文件
* 使用AudioRecord录制的直接生成的是PCM(脉冲编码调制)文件(IO操作),该文件可以用AudioTrack进行播放但是普通播放器可能播放不来
* 可以添加文件头并修改为wav格式,就成了无损音频,大部分手机音乐播放器可以播放,可以使用MediaExtractor获取文件信息buffer
* 1、AudioRecord录制的pcm是无损的,Mediacodec使用的是H264编码解码,H264是MPEG-4标准所定义的最新编码格式,同时也是技术含量最高、代表最新技术水品的编码格式之一,标准格式是H.264,H264格式是经过有损压缩的,但在技术上尽可能做到了降低存储体积的情况下获得较好图像质量和低宽带图像快速传输。
* 文件格式、视频封装格式、视频编码方式区别:https://blog.csdn.net/electrombile/article/details/51067344


//不同文件保存方式情况下,各播放器播放测试
* muxer保存时不添加后缀名:muxer_video 手机选择视频打开可以播放;电脑使用windowmediaplayer打开,弹框:无法识别扩展名但仍然可以播放;muxer_audio 手机选择音频无法打开,用视频播放器可以打开;电脑使用windowmediaplayer打开,弹框:无法识别扩展名但仍然可以播放。
* file_audio  手机选择音频无法打开,用视频播放器可以打开;电脑使用windowmediaplayer打开,弹框:无法识别扩展名但仍然可以播放。
* 分离出音频时,如果给音频添加后缀名aac或者买a,则都可以用音频打开

* mediaFormat 的信息打印参见 https://www.jianshu.com/p/180f81390b58


private static finical String SDCARD_PATH = Environment.getExternalStorageDirectory().getPath() + "/Pictures/Screenshots";

//将视频分离为音频muxer_audio.m4a和无声视频muxer_vido.mp4
private void splitAudioAndVideo() {
    MediaExtractor mediaExtractor = new MediaExtractor();
    MediaFormat audioTrackFormat = null;
    MediaFormat videoTrackFormat = null;
    int audioTrackIndex = -1;
    int vidoeTrackIndex = -1;
    try {
        mediaExtractor.setDataSource(SDCARD_PATH + "/input.mp4");
        int trackCount = mediaExtractor.getTrackCount();
        //分别获取音轨视轨格式和对应的轨道编号
        for(int i = 0; i <trackCount; i++) {
            MediaFormat trackFormat = mediaExtractor.getTrackFormat(i);
            String keyMime = trackFormat.getString(MediaFormat.KEY_MIME);
            if(keyMime.startsWith("video")) {
                videoTrackFormat = trackFormat;
                vidoeTrackIndex = i;
            } else if (keyMime.startsWith("audio")) {
                audioTrackFormat = trackFormat;
                audioTrackIndex = i;
            }
        }

        //读取input.mp4中的的音频数据,并用mediacodec进行编码生成音频,生成的muxer_audio文件,不加后缀名的话,无法使用手机的音乐播放器打开,可以用视频播放器打开,电脑上用windowsmediaplayer也可以打开
        //如果生成的音频是加后缀名的如muxer_audio.aac或者muxer_audio.m4a,可以用手机的音乐播放器软件打开

        mediaExtractor.selectTrack(audioTrackIndex);
        MediaMuxer audioMediaMuxer = new MediaMuxer(SDCARD_PATH + "/muxer_audio.m4a", MediaMuxer.OutputFormat.MUXER_OUTPUT_MPEG_4);
        int trackIndex = audioMediaMuxer.addTrack(audioTrackFormat);
        audioMediaMuxer.start();

        ByteBuffer maxInputByteBuffer = ByteBuffer.allocate(videoTrackFormat.getlnteger(MediaFormat.KEY_MAX_INPUT_SIZE));
        MediaCodec.Bufferlnfo audioBufferlnfo = new MediaCodec.BufferlnfoO;
        while (true) {
            int readSampleData = mediaExtractor.readSampleData(m axInputByteBuffer, 0);
            if (readSampleData < 0) {
                mediaExtractor.unselectTrack(audioTracklndex);
                break;
            }
            audioBufferlnfo.size = readSampleData;
            audioBufferlnfo.offset = 0;
            audioBufferlnfo.presentationTimeUs = mediaExtractor.getSampleTime();

            audioBufferlnfo.flags = mediaExtractor.getSampleFlags();
            audioMediaMuxer.writeSampleData(tracklndex, maxInputByteBuffer, audioBufferlnfo);
            mediaExtractor.advance();
        }
        if (audioMediaMuxer != null) {
            audioMediaMuxer.stop();
            audioMediaMuxer.release();
        }


        //读取input.mp4中的视频数据,并用mediacodec进行编码成不带声音的视频
        mediaExtractor.selectTrack(videoTrackIndex);
        MediaMuxer audioMediaMuxer = new MediaMuxer(SDCARD_PATH + "/muxer_video.mp4", MediaMuxer.OutputFormat.MUXER_OUTPUT_MPEG_4);
        int i = videoMediaMuxer.addTrack(videoTrackFormat);
        videoMediaMuxer.start();
        ByteBuffer allocate = ByteBuffer.allocate(videoTrackFormat.getlnteger(MediaFormat.KEY_MAX_INPUT_SIZE));
        MediaCodec.Bufferlnfo bufferlnfo = new MediaCodec.Bufferlnfo();
        while (true) {
            int readSampleData = mediaExtractor.readSampleData(allocate, 0);
            if (readSampleData < 0) {
                mediaExtractor.unselectTrack( videoTrackIndex);
                break;
            }
            bufferlnfo.flags = mediaExtractor.getSampleFlags();
            bufferlnfo.presentationTimeUs = mediaExtractor.getSampleTime();
            bufferlnfo.offset = 0;
            bufferlnfo.size = readSampleData;
            videoMediaMuxer.writeSampleData( i, allocate, bufferlnfo);
            mediaExtractor.advance();
        }
        videoMediaMuxer.stop();
        videoMediaMuxer.release();
        mediaExtractor.release();
        Toast.makeText(this,"音频和视频分离成功", Toast_ENGTH_SHORT).show();
    } catch (lOException e) {
        e.printStackTraceO;
    }
}

//将分离的音频音频muxer_audio.m4a和无声视频muxer_vido.mp4合成为原来的视频
private void combineAudioAndVideo() {
    try {
        //获取音频格式及音轨
        MediaExtractor audioMediaExtractor = new MediaExtractor();
        audioMediaExtractor.setDataSource(SDCARD_PATH + "/muxer_audio.m4a");
        int audioTrackConut = audioMediaExtractor.getTrackCount();
        MediaFormat audioTrackFormat = null;
        int audioExtractorIndex = 0;
        for (int i = 0; i <audioTrackConut; i++) {
        audioTrackFormat = audioMediaExtractor.getTrackFormat(i);
            String keyMime = audioMediaExtractor.getString(MediaFormat.KEY_MIME);
            if (keyMime.startsWith("audio")) {
                audioExtractorIndex = i;
            }
        }
        audioMediaExtractor.selectTrack(audioExtractorIndex);

        //获取视频格式及视轨
        MediaExtractor videoMediaExtractor = new MediaExtractor();
        videoMediaExtractor.setDataSource(SDCARD_PATH + "/muxer_video.mp4");
        int videoTrackConut = videoMediaExtractor.getTrackCount();
        MediaFormat videoTrackFormat = null;
        int videoExtractorIndex = 0;
        for (int i = 0; i <videoTrackConut; i++) {
        videoTrackFormat = videoMediaExtractor.getTrackFormat(i);
            String keyMime = videoMediaExtractor.getString(MediaFormat.KEY_MIME);
            if (keyMime.startsWith("video")) {
                videoExtractorIndex = i;
            }
        }
        videoMediaExtractor.selectTrack(videoExtractorIndex);

        //开始进行混合
        MediaMuxer mediaMuxer = new MediaMuxer(SDCARD_PATH + "/muxer_comnine.mp4", MediaMuxer.OutputFormat.MUXER_OUTPUT_MPEG_4);
        int audioTrackIndex = mediaMuxer.addTrack(audioTrackFormat);
        int videoTrackIndex = mediaMuxer.addTrack(videoTrackFormat);
        audioMediaMuxer.start();

        //视频的音轨、视轨分辨处理,先将音轨混合到视频文件中,混合结束后再混合视轨
        //也可以音轨一帧视轨一帧交错混合,MediaMuxer在writeSampleData时,传入的轨道编号和格式信息不同,就会将不同的数据混合到相应的轨道

        //交错处理
        ByteBuffer audioByteBuffer = ByteBuffer.allocate(audioTrackFormat.getlnteger(MediaFormat.KEY_MAX_INPUT_SIZE));
        ByteBuffer videoByteBuffer = ByteBuffer.allocate(videoTrackFormat.getlnteger(MediaFormat.KEY_MAX_INPUT_SIZE));

        MediaCodec.BufferInfo audioBufferInfo = MediaCodec.BufferInfo();
        MediaCodec.BufferInfo videoBufferInfo = MediaCodec.BufferInfo();

        while(true) {
        int readSampleDataAudio = audioMediaExtractor.readSampleData(audioByteBuffer,0);
            if(readSampleDataAudio > 0) {
                audioBufferlnfo.size = readSampleDataAudio;
                audioBufferlnfo.flags = audioMediaExtractor.getSampleFlags();
                audioBufferlnfo.offset = 0;
                audioBufferlnfo.presentationTimeUs = audioMediaExtractor.getSampleTime();
                audioMediaMuxer.writeSampleData(audioTrackIndex, audioByteBuffer, audioBufferlnfo);
                audioMediaExtractor.advance();
            }

            int readSampleDataVideo = videoMediaExtractor.readSampleData(videoByteBuffer,0);
            if(readSampleDataVideo > 0) {
                videoBufferInfo.size = readSampleDataVideo;
                videoBufferInfo.flags = videoMediaExtractor.getSampleFlags();
                videoBufferInfo.offset = 0;
                videoBufferInfo.presentationTimeUs = videoMediaExtractor.getSampleTime();
                videoBufferInfo.writeSampleData(videoTrackIndex, videoByteBuffer, videoBufferInfo);
                videoMediaExtractor.advance();
            }

            if (readSampleDataAudio < 0 && readSampleDataVideo < 0) {
                audioMediaExtractor.unselectTrack(videoExtractorIndex);
                videoMediaExtractor.unselectTrack(videoExtractorIndex);
                break;
            }
        }

        mediaMuxer.stop();
        mediaMuxer.release();

        audioMediaExtractor.release();
        videoMediaExtractor.release();

        Toast.makeText(this,"分离的视频音频合成完成", Toast.LENGTH_SHOT).show();

    } catch(IOException e) {
        e.printStackTrace();
    }
}


//同时分离出音频和视频file,音频添加头可以播放,视频不能播放
private void exactorMediaToFile() {
    FileOutputStream vos = null;
    FileOutputStream aos = null;
    try {
        File videoFile = new File(SDCARD_PATH, "file_video.mp4");
        if(!videoFile.exists()) {
            videoFile.createNewFile();
        }
        File audioFile = new File(SDCARD_PATH, "file_audio.acc");
        vos = new FileOutputStream(videoFile);
        aos = new FileOutputStream(audioFile);

        mediaExtractor.setDataSource(SDCARD_PATH+"/input.mp4");
        int trackCount = mediaExtractor.getTrackCount;
        int audioTrackIndex = -1;
        int vidoeTrackIndex = -1;

        for(int i = 0; i <trackCount; i++) {
            MediaFormat trackFormat = mediaExtractor.getTrackFormat(i);
            String mineType = trackFormat.getString(MediaFormat.KEY_MIME);
            if(mineType.startsWith("video")) {
                vidoeTrackIndex = i;
            }
            if (mineType.startsWith("audio")) {
                audioTrackIndex = i;
            }

            ByteBuffer byteBuffer = ByteBuffer.allocate(500*1024);

            mediaExtractor.selectTrack(videoTrackIndex);
            while(true) {
                int readSampleCount = mediaExtractor.readSampleData(byteBuffer,0);
                if(readSampleCount <0 ) {
                    break;
                }
                byte[] buffer = new byte[readSampleCount];
                byteBuffer.get(buffer);
                vos.write(buffer);
                byteBuffer.clear();
                mediaExtractor.advance();
            }

            mediaExtractor.selectTrack(audioTrackIndex);
            while(true) {
                int readSampleCount = mediaExtractor.readSampleData(byteBuffer,0);
                if(readSampleCount <0 ) {
                    break;
                }
                byte[] buffer = new byte[readSampleCount];
                byteBuffer.get(buffer);

                //此处添加文件头,音频才可以播放,直接write buffer的文件不能播放
                byte[] newbuff = new byte[readSampleCount + 7];
                addADTStoPacket(newbuff, readSampleCount+7);
                System.arraycopy(buffer, 0, newbuff, 7, readSampleCount);

                aos.write(newbuff);
                byteBuffer.clear();
                mediaExtractor.advance();
            }

            Toast.makeText(this,"分离视频通过IO保存完成", Toast.LENGTH_SHOT).show();

        } catch(Exception e) {
            e.printStackTrace();
        } finally {
            mediaExtractor.release();
            try {
                vos.close();
            } catch(Exception e) {
                e.printStackTrace();
            }
        }
    }
}

private static void addADTStoPacket(byte[] packet, int packetLen) {

    int profile = 2; // AAC LC
    int freqIdx = 3; // 8表示16000,取待定
    int chanCfg = 1; // 音频声道数

    // fill in ADTS data
    packet[0] = (byte) 0xFF;
    packet[1] = (byte) 0xF1;
    packet[2] = (byte) (((profile - 1) << 6) + (freqIdx << 2) + (chanCfg >> 2));
    packet[3] = (byte) (((chanCfg & 3) << 6) + (packetLen >> 11));
    packet[4] = (byte) ((packetLen & 0x7FF) >> 3);
    packet[5] = (byte) (((packetLen & 7) << 5) + 0x1F);

}

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java本身不提供对视频和音频的处理功能,但可以使用第三方库来实现去除视频文件中的声音。常用的库是Xuggler和JAVE(Java Audio Video Encoder),它们都支持通过Java代码调用FFmpeg。以下是使用Xuggler的示例代码: ```java import com.xuggle.mediatool.IMediaReader; import com.xuggle.mediatool.IMediaWriter; import com.xuggle.mediatool.ToolFactory; import com.xuggle.xuggler.ICodec; public class RemoveAudioFromVideo { public static void main(String[] args) { String inputFilePath = "input.mp4"; String outputFilePath = "output.mp4"; IMediaReader mediaReader = ToolFactory.makeReader(inputFilePath); IMediaWriter mediaWriter = ToolFactory.makeWriter(outputFilePath, mediaReader); mediaReader.addListener(mediaWriter); mediaWriter.setForceInterleave(true); mediaWriter.addVideoStream(0, 0, ICodec.ID.CODEC_ID_H264, mediaReader.getContainer().getStream(0).getFrameRate().getDouble()); mediaWriter.addAudioStream(1, 0, 0, 0, 0); while (mediaReader.readPacket() == null) ; } } ``` 这段代码使用Xuggler读取输入文件,并将视频流写入到输出文件中,同时添加一个空音频流,从而达到去除视频文件中声音的目的。需要注意的是,Xuggler已经停止维护,如果需要更高级的功能,建议使用FFmpeg的Java接口或其他第三方库。 ### 回答2: Java本身没有提供直接去除视频文件中声音的功能,需要借助外部库或工具来实现此功能。FFmpeg是一个功能强大的音视频处理工具,可以用于处理和编辑音视频文件,包括去除视频中的声音。然而,Java并不直接集成FFmpeg,因此无法直接使用FFmpeg去除视频文件中的声音。 要使用Java去除视频文件中的声音,可以借助其他的Java库或框架来实现。一种方法是使用JavaCV库,它是一个基于Java的OpenCV和FFmpeg的接口库,可以让Java程序调用FFmpeg的功能。在使用JavaCV时,你可以调用FFmpeg的命令行参数来去除视频文件中的声音。 另一种方法是使用Java自带的音视频处理库,如javax.sound.sampled库来处理音频,然后使用Java图形库如JavaFX来处理视频。可以使用这些库读取原始的视频文件和音频文件,然后分离音频轨道和视频轨道,再将分离出的视频轨道合成成一个新的视频文件,从而达到去除视频文件中声音的效果。 总结来说,Java本身没有直接提供去除视频文件中声音的功能,但可以借助其他的Java库或工具来实现此功能,如JavaCV库或javax.sound.sampled库等。 ### 回答3: 在Java中,不使用FFmpeg去除视频文件中的声音是有一定的难度的。因为Java自带的库并没有提供直接处理视频文件的功能。但是,仍然可以通过其他方式来实现去除视频文件中声音的功能。 一种方法是使用Java提供的ProcessBuilder类来执行命令行操作。可以使用该类执行FFmpeg的命令行命令,通过命令行参数来实现去除视频文件中声音的功能。比如可以调用FFmpeg的命令行工具,使用“-an”参数来指定不提取音频流。这样执行命令后,会生成一个没有声音的新视频文件。 另一种方法是使用第三方Java库来处理视频文件。虽然Java本身没有支持直接处理视频文件的功能,但是有一些第三方Java库可以实现视频处理的功能,并可以实现去除声音的操作。例如,可以使用Xuggler库或者JavaCV库来处理视频文件。这些库提供了对FFmpeg的封装,可以在Java中调用FFmpeg的功能,包括去除视频文件中的声音。 无论使用哪种方法,都需要先确保系统中已经安装了FFmpeg,并正确设置了环境变量。然后,在Java代码中调用相应的命令行或者第三方库的函数,传入视频文件路径和相应的参数,即可实现去除视频文件中声音的功能。 需要注意的是,由于FFmpeg是一个强大的多媒体处理工具,其命令行参数众多,使用起来可能需要一定的学习和理解成本。同时,对于大文件或者复杂的视频文件,去除声音可能会消耗较长的时间和系统资源。因此,在实际应用中,需要根据具体情况选择最合适的方法来去除视频文件中的声音。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值