利用MediaExtractor和MediaMuxer实现视频剪切

2023-11-04 09:10

本文主要是介绍利用MediaExtractor和MediaMuxer实现视频剪切,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

客户要在Android手机上做个能视频剪切的app,由于视频源只是MP4,所以就想到了用MediaExtractor和MediaMuxer来实现功能,直接上代码。

public class VideoDecoder {private final static String TAG = "VideoDecoder";private MediaCodec mediaDecoder;private MediaExtractor mediaExtractor;private MediaFormat mediaFormat;private MediaMuxer mediaMuxer;private String mime = null;public boolean decodeVideo(String url, long clipPoint, long clipDuration) {int videoTrackIndex = -1;int audioTrackIndex = -1;int videoMaxInputSize = 0;int audioMaxInputSize = 0;int sourceVTrack = 0;int sourceATrack = 0;long videoDuration, audioDuration;//创建分离器mediaExtractor = new MediaExtractor();try {//设置文件路径mediaExtractor.setDataSource(url);//创建合成器mediaMuxer = new MediaMuxer(url.substring(0, url.lastIndexOf(".")) + "_output.mp4", OutputFormat.MUXER_OUTPUT_MPEG_4);} catch (Exception e) {Log.e(TAG, "error path" + e.getMessage());}//获取每个轨道的信息for (int i = 0; i < mediaExtractor.getTrackCount(); i++) {try {mediaFormat = mediaExtractor.getTrackFormat(i);mime = mediaFormat.getString(MediaFormat.KEY_MIME);if (mime.startsWith("video/")) {sourceVTrack = i;int width = mediaFormat.getInteger(MediaFormat.KEY_WIDTH);int height = mediaFormat.getInteger(MediaFormat.KEY_HEIGHT);videoMaxInputSize = mediaFormat.getInteger(MediaFormat.KEY_MAX_INPUT_SIZE);videoDuration = mediaFormat.getLong(MediaFormat.KEY_DURATION);//检测剪辑点和剪辑时长是否正确 if (clipPoint >= videoDuration) {Log.e(TAG, "clip point is error!");return false;}if ((clipDuration != 0) && ((clipDuration + clipPoint) >= videoDuration)) {Log.e(TAG, "clip duration is error!");return false;}Log.d(TAG, "width and height is " + width + " " + height+ ";maxInputSize is " + videoMaxInputSize+ ";duration is " + videoDuration);//向合成器添加视频轨videoTrackIndex = mediaMuxer.addTrack(mediaFormat);}else if (mime.startsWith("audio/")) {sourceATrack = i;int sampleRate = mediaFormat.getInteger(MediaFormat.KEY_SAMPLE_RATE);int channelCount = mediaFormat.getInteger(MediaFormat.KEY_CHANNEL_COUNT);audioMaxInputSize = mediaFormat.getInteger(MediaFormat.KEY_MAX_INPUT_SIZE);audioDuration = mediaFormat.getLong(MediaFormat.KEY_DURATION);Log.d(TAG, "sampleRate is " + sampleRate+ ";channelCount is " + channelCount+ ";audioMaxInputSize is " + audioMaxInputSize+ ";audioDuration is " + audioDuration);//添加音轨audioTrackIndex = mediaMuxer.addTrack(mediaFormat);}Log.d(TAG, "file mime is " + mime);} catch (Exception e) {Log.e(TAG, " read error " + e.getMessage());}}//分配缓冲ByteBuffer inputBuffer = ByteBuffer.allocate(videoMaxInputSize);//根据官方文档的解释MediaMuxer的start一定要在addTrack之后mediaMuxer.start();//视频处理部分mediaExtractor.selectTrack(sourceVTrack);MediaCodec.BufferInfo videoInfo = new MediaCodec.BufferInfo();videoInfo.presentationTimeUs = 0;long videoSampleTime;//获取源视频相邻帧之间的时间间隔。(1){mediaExtractor.readSampleData(inputBuffer, 0);//skip first I frameif (mediaExtractor.getSampleFlags() == MediaExtractor.SAMPLE_FLAG_SYNC)mediaExtractor.advance();mediaExtractor.readSampleData(inputBuffer, 0);long firstVideoPTS = mediaExtractor.getSampleTime();mediaExtractor.advance();mediaExtractor.readSampleData(inputBuffer, 0);long SecondVideoPTS = mediaExtractor.getSampleTime();videoSampleTime = Math.abs(SecondVideoPTS - firstVideoPTS);Log.d(TAG, "videoSampleTime is " + videoSampleTime);}//选择起点mediaExtractor.seekTo(clipPoint, MediaExtractor.SEEK_TO_PREVIOUS_SYNC);while (true) {int sampleSize = mediaExtractor.readSampleData(inputBuffer, 0);if (sampleSize < 0) {//这里一定要释放选择的轨道,不然另一个轨道就无法选中了mediaExtractor.unselectTrack(sourceVTrack);break;}int trackIndex = mediaExtractor.getSampleTrackIndex();//获取时间戳long presentationTimeUs = mediaExtractor.getSampleTime();//获取帧类型,只能识别是否为I帧int sampleFlag = mediaExtractor.getSampleFlags();Log.d(TAG, "trackIndex is " + trackIndex+ ";presentationTimeUs is " + presentationTimeUs+ ";sampleFlag is " + sampleFlag+ ";sampleSize is " + sampleSize);//剪辑时间到了就跳出if ((clipDuration != 0) && (presentationTimeUs > (clipPoint + clipDuration))) {mediaExtractor.unselectTrack(sourceVTrack);break;}mediaExtractor.advance();videoInfo.offset = 0;videoInfo.size = sampleSize;videoInfo.flags = sampleFlag;mediaMuxer.writeSampleData(videoTrackIndex, inputBuffer, videoInfo);videoInfo.presentationTimeUs += videoSampleTime;//presentationTimeUs;}//音频部分mediaExtractor.selectTrack(sourceATrack);MediaCodec.BufferInfo audioInfo = new MediaCodec.BufferInfo();audioInfo.presentationTimeUs = 0;long audioSampleTime;//获取音频帧时长{mediaExtractor.readSampleData(inputBuffer, 0);//skip first sampleif (mediaExtractor.getSampleTime() == 0)mediaExtractor.advance();mediaExtractor.readSampleData(inputBuffer, 0);long firstAudioPTS = mediaExtractor.getSampleTime();mediaExtractor.advance();mediaExtractor.readSampleData(inputBuffer, 0);long SecondAudioPTS = mediaExtractor.getSampleTime();audioSampleTime = Math.abs(SecondAudioPTS - firstAudioPTS);Log.d(TAG, "AudioSampleTime is " + audioSampleTime);}mediaExtractor.seekTo(clipPoint, MediaExtractor.SEEK_TO_CLOSEST_SYNC);while (true) {int sampleSize = mediaExtractor.readSampleData(inputBuffer, 0);if (sampleSize < 0) {mediaExtractor.unselectTrack(sourceATrack);break;}int trackIndex = mediaExtractor.getSampleTrackIndex();long presentationTimeUs = mediaExtractor.getSampleTime();Log.d(TAG, "trackIndex is " + trackIndex+ ";presentationTimeUs is " + presentationTimeUs);if ((clipDuration != 0) && (presentationTimeUs > (clipPoint + clipDuration))) {mediaExtractor.unselectTrack(sourceATrack);break;}mediaExtractor.advance();audioInfo.offset = 0;audioInfo.size = sampleSize;mediaMuxer.writeSampleData(audioTrackIndex, inputBuffer, audioInfo);audioInfo.presentationTimeUs += audioSampleTime;//presentationTimeUs;}//全部写完后释放MediaMuxer和MediaExtractormediaMuxer.stop();mediaMuxer.release();mediaExtractor.release();mediaExtractor = null;return true;}
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
  • 131
  • 132
  • 133
  • 134
  • 135
  • 136
  • 137
  • 138
  • 139
  • 140
  • 141
  • 142
  • 143
  • 144
  • 145
  • 146
  • 147
  • 148
  • 149
  • 150
  • 151
  • 152
  • 153
  • 154
  • 155
  • 156
  • 157
  • 158
  • 159
  • 160
  • 161
  • 162
  • 163
  • 164
  • 165
  • 166
  • 167
  • 168
  • 169
  • 170
  • 171
  • 172
  • 173
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
  • 131
  • 132
  • 133
  • 134
  • 135
  • 136
  • 137
  • 138
  • 139
  • 140
  • 141
  • 142
  • 143
  • 144
  • 145
  • 146
  • 147
  • 148
  • 149
  • 150
  • 151
  • 152
  • 153
  • 154
  • 155
  • 156
  • 157
  • 158
  • 159
  • 160
  • 161
  • 162
  • 163
  • 164
  • 165
  • 166
  • 167
  • 168
  • 169
  • 170
  • 171
  • 172
  • 173

这里要对(1)进行一下解释,如果视频中带了B帧,源视频时间戳会按照播放顺序来打,这样通过读每帧获取到的时间戳就不是一直增长的,MediaMuxer的writeSampleData函数在写每帧的时间戳时如果时间戳不是增长顺序就会报错,帧率也无法通过MediaFormat.KEY_FRAME_RATE来获取,也可以通过查找MP4文件的stsz关键字来获取总帧数,进而算出帧率,来按帧率给每一帧打时间戳,但是stsz一般都在文件的后面而且含有音视频的文件会有两个stsz,如果文件很大,查找起来就会很慢,所以这里我用了个偷懒的办法,用I帧后面两帧的时间戳绝对差值来当做每帧的时间,然后+=这个时间来为写入的帧打时间戳,虽然时间戳不是源视频的顺序了,但并不影响解码。 
还有一点就是unselectTrack这个函数,用selectTrack选择轨道后,如果要切换轨道一定要调用unselectTrack函数来释放轨道,不然读的还是之前的轨道。

这篇关于利用MediaExtractor和MediaMuxer实现视频剪切的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/344736

相关文章

Python实现批量CSV转Excel的高性能处理方案

《Python实现批量CSV转Excel的高性能处理方案》在日常办公中,我们经常需要将CSV格式的数据转换为Excel文件,本文将介绍一个基于Python的高性能解决方案,感兴趣的小伙伴可以跟随小编一... 目录一、场景需求二、技术方案三、核心代码四、批量处理方案五、性能优化六、使用示例完整代码七、小结一、

Java实现将HTML文件与字符串转换为图片

《Java实现将HTML文件与字符串转换为图片》在Java开发中,我们经常会遇到将HTML内容转换为图片的需求,本文小编就来和大家详细讲讲如何使用FreeSpire.DocforJava库来实现这一功... 目录前言核心实现:html 转图片完整代码场景 1:转换本地 HTML 文件为图片场景 2:转换 H

C#使用Spire.Doc for .NET实现HTML转Word的高效方案

《C#使用Spire.Docfor.NET实现HTML转Word的高效方案》在Web开发中,HTML内容的生成与处理是高频需求,然而,当用户需要将HTML页面或动态生成的HTML字符串转换为Wor... 目录引言一、html转Word的典型场景与挑战二、用 Spire.Doc 实现 HTML 转 Word1

C#实现一键批量合并PDF文档

《C#实现一键批量合并PDF文档》这篇文章主要为大家详细介绍了如何使用C#实现一键批量合并PDF文档功能,文中的示例代码简洁易懂,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言效果展示功能实现1、添加文件2、文件分组(书签)3、定义页码范围4、自定义显示5、定义页面尺寸6、PDF批量合并7、其他方法

SpringBoot实现不同接口指定上传文件大小的具体步骤

《SpringBoot实现不同接口指定上传文件大小的具体步骤》:本文主要介绍在SpringBoot中通过自定义注解、AOP拦截和配置文件实现不同接口上传文件大小限制的方法,强调需设置全局阈值远大于... 目录一  springboot实现不同接口指定文件大小1.1 思路说明1.2 工程启动说明二 具体实施2

Python实现精确小数计算的完全指南

《Python实现精确小数计算的完全指南》在金融计算、科学实验和工程领域,浮点数精度问题一直是开发者面临的重大挑战,本文将深入解析Python精确小数计算技术体系,感兴趣的小伙伴可以了解一下... 目录引言:小数精度问题的核心挑战一、浮点数精度问题分析1.1 浮点数精度陷阱1.2 浮点数误差来源二、基础解决

Java实现在Word文档中添加文本水印和图片水印的操作指南

《Java实现在Word文档中添加文本水印和图片水印的操作指南》在当今数字时代,文档的自动化处理与安全防护变得尤为重要,无论是为了保护版权、推广品牌,还是为了在文档中加入特定的标识,为Word文档添加... 目录引言Spire.Doc for Java:高效Word文档处理的利器代码实战:使用Java为Wo

Java实现远程执行Shell指令

《Java实现远程执行Shell指令》文章介绍使用JSch在SpringBoot项目中实现远程Shell操作,涵盖环境配置、依赖引入及工具类编写,详解分号和双与号执行多指令的区别... 目录软硬件环境说明编写执行Shell指令的工具类总结jsch(Java Secure Channel)是SSH2的一个纯J

使用Python实现Word文档的自动化对比方案

《使用Python实现Word文档的自动化对比方案》我们经常需要比较两个Word文档的版本差异,无论是合同修订、论文修改还是代码文档更新,人工比对不仅效率低下,还容易遗漏关键改动,下面通过一个实际案例... 目录引言一、使用python-docx库解析文档结构二、使用difflib进行差异比对三、高级对比方

深度解析Python中递归下降解析器的原理与实现

《深度解析Python中递归下降解析器的原理与实现》在编译器设计、配置文件处理和数据转换领域,递归下降解析器是最常用且最直观的解析技术,本文将详细介绍递归下降解析器的原理与实现,感兴趣的小伙伴可以跟随... 目录引言:解析器的核心价值一、递归下降解析器基础1.1 核心概念解析1.2 基本架构二、简单算术表达