Android(java方法)上实现mp4的分割和拼接 (二)
这节谈一下如何在android上实现mp4文件的高效率切割。
业务需求举例:把一段2分钟的mp4文件切割出00:42 至 01:16这段时间的视频,要求足够短的执行时间和尽量少的误差。
分析:mp4Parser只能在关键帧切割,比如,在00:40和00:45分别存在一个可切割关键帧,那么切割视频的头和尾,都应该选择短切割。然后获取到误差的视频短,如果这个误差大于0.5S,用FFmpeg进行一帧一帧编解码切割文件。这样最多会有三段mp4文件,再次将这三段mp4拼接起来就可以了。
下面直接上关键代码,这些代码在PC上新建一个java工程也可以实现。
1.切割文件方法:
/**
需要使用isoviewer-1.0-RC-27包
返回值是目标mp4的开头和结尾时刻
**/
- public static double[] startTrim(File src, File dst, int startMs, int endMs) throws IOException {
- Movie movie = MovieCreator.build(src.getAbsolutePath());
- List<Track> tracks = movie.getTracks();
- movie.setTracks(new LinkedList<Track>());
- double startTime = startMs/1000;
- double endTime = endMs/1000;
- boolean timeCorrected = false;
- // Here we try to find a track that has sync samples. Since we can only start decoding
- // at such a sample we SHOULD make sure that the start of the new fragment is exactly
- // such a frame
- for (Track track : tracks) {
- if (track.getSyncSamples() != null && track.getSyncSamples().length > 0) {
- if (timeCorrected) {
- throw new RuntimeException("The startTime has already been corrected by another track with SyncSample. Not Supported.");
- }
- //true,false表示短截取;false,true表示长截取
- startTime = correctTimeToSyncSample(track, startTime, true);
- endTime = correctTimeToSyncSample(track, endTime, false);
- timeCorrected = true;
- }
- }
- int x = 0;
- for (Track track : tracks) {
- long currentSample = 0;
- double currentTime = 0;
- long startSample = -1;
- long endSample = -1;
- x++;
- for (int i = 0; i < track.getDecodingTimeEntries().size(); i++) {
- TimeToSampleBox.Entry entry = track.getDecodingTimeEntries().get(i);
- for (int j = 0; j < entry.getCount(); j++) {
- // entry.getDelta() is the amount of time the current sample covers.
- if (currentTime <= startTime) {
- // current sample is still before the new starttime
- startSample = currentSample;
- }
- if (currentTime <= endTime) {
- // current sample is after the new start time and still before the new endtime
- endSample = currentSample;
- } else {
- // current sample is after the end of the cropped video
- break;
- }
- currentTime += (double) entry.getDelta() / (double) track.getTrackMetaData().getTimescale();
- currentSample++;
- }
- }
- movie.addTrack(new CroppedTrack(track, startSample, endSample));
- break;
- }
- Container container = new DefaultMp4Builder().build(movie);
- if (!dst.exists()) {
- dst.createNewFile();
- }
- FileOutputStream fos = new FileOutputStream(dst);
- FileChannel fc = fos.getChannel();
- container.writeContainer(fc);
- fc.close();
- fos.close();
- double[] doubleArray = new double[2] ;
- doubleArray[0] = startTime;
- doubleArray[1] = endTime;
- return doubleArray;
- }
2.ffmpeg切割方法,需要jni实现。稍后补充
- public String getMp4ByFFmpeg(double mTimeStart,double mTimeEnd,String videoPath){
- try{
- String mFinalVideoPath = videoPath;
- int audioChannels = 2;
- FFmpegRecorder recorder = new FFmpegRecorder(
- mFinalVideoPath, RecorderConfig.TARGET_VIDEO_WIDTH,
- RecorderConfig.TARGET_VIDEO_HEIGHT, audioChannels);
- RecorderConfig.setRecorderConfig(recorder, RecorderConfig.CONFIG_TYPE_MPEG4_HIGH);
- int totalFrames = 0;
- FFmpegGrabber grabber = FFmpegGrabber.createDefault(mPath);
- grabber.setSquareSize(RecorderConfig.TARGET_VIDEO_WIDTH);
- int degree = VideoFileUtil.getRotate(mPath);
- grabber.setOrientation(degree);
- grabber.start();
- if (mTimeStart > 0) {
- grabber.setTimestamp((long)mTimeStart);
- }
- totalFrames = grabber.getLengthInFrames();
- VideoClip mFinalClip = new VideoClip();
- mFinalClip.mIsFromLocal = true;
- mFinalClip.mHeight = RecorderConfig.TARGET_VIDEO_HEIGHT;
- mFinalClip.mWidth = RecorderConfig.TARGET_VIDEO_WIDTH;
- recorder.setAudioChannels(grabber.getAudioChannels());
- recorder.setSampleRate(grabber.getSampleRate());
- recorder.setAudioCodec(avcodec.AV_CODEC_ID_AAC);
- recorder.setFrameRate(FFmpegRecorder.DEFAULT_FRAME_RATE);
- recorder.setVideoCodec(avcodec.AV_CODEC_ID_MPEG4);
- recorder.start();
- mFinalClip.mOrientation = 0;
- mFinalClip.mFrameRate = (int) recorder.getFrameRate();
- mFinalClip.mSampleRate = recorder.getSampleRate();
- mFinalClip.mAudioBitrate = recorder.getAudioBitrate();
- mFinalClip.mAudioChannels = recorder.getAudioChannels();
- Frame grabbedFrame = new Frame();
- int j = 0;
- boolean videoTimeout = false;
- boolean audioTimeout = false;
- while (grabber.grabFrame(grabbedFrame)) {
- long i = grabber.getTimestamp();
- long k = grabber.getFrameNumber();
- if (videoTimeout && audioTimeout) {
- break;
- }
- if (grabbedFrame.hasVideoFrame()) {
- int progress = 100 * (int) (i - mTimeStart) / mTotalTimeSpan;
- publishProgress(progress);
- }
- if (i > mTimeEnd) {
- if (grabbedFrame.hasAudioFrame()) {
- audioTimeout = true;
- }
- if (grabbedFrame.hasVideoFrame()) {
- videoTimeout = true;
- }
- continue;
- }
- grabbedFrame.setTimeStamp((long)(i - mTimeStart));
- recorder.recordFrameNoException(grabbedFrame);
- SLog.v(TAG, "record image at {}, #{}", i, k);
- j++;
- }
- grabbedFrame.releaseNativeAllocation();
- grabber.stop();
- grabber.release();
- recorder.stop();
- recorder.release();
- mFinalClip.mClipPath = mFinalVideoPath;
- mFinalClip.mDuration = (long) (MP4ParserUtil.getDuration(mFinalVideoPath) * 1000);
- mFinalClip.mTargetMills = mFinalClip.mDuration;
- return mFinalVideoPath;
- } catch (Exception ex) {
- return null;
- }
- }
3.拼接三段视频代码
- public boolean newClipMethod(String dstFile,String srcFile){
- try {
- double[] results = ClipMp4Util.startTrim(new File(dstFile),new File(srcFile),mTimeStart,mTimeEnd);
- if(results == null){
- return false;
- }
- Log.d("","newClipMethod-->results[0]-mTimeStart"+results[0]+" "+mTimeStart/1000);
- Log.d("","newClipMethod-->mTimeEnd-results[1]"+mTimeEnd/1000+" "+results[1]);
- //下面是短截取然后拼接的逻辑
- if(results[0]-mTimeStart/1000>GAP){
- String startMp4 = <span style="font-family: Arial, Helvetica, sans-serif;">getMp4ByFFmpeg(</span><span style="font-family: Arial, Helvetica, sans-serif;">mTimeStart,results[0]*1000,begin);</span>
- }
- if(mTimeEnd/1000-results[1]>GAP){
- String endMp4 = <span style="font-family: Arial, Helvetica, sans-serif;">getMp4ByCode(</span><span style="font-family: Arial, Helvetica, sans-serif;">results[1]*吧1000,mTimeEnd,end);</span>
- }
- String[] videos = new String[3];
- videos[0] = begin;
- videos[1] = dst;
- videos[2] = end;
- appendVideo(videos);
- } catch (Exception e) {
- //如果不是同一格式的视频,这里合成会报错,直接返回中间视频.所以长视频选取长误差的方式,前后都多截取一段
- Log.d("","new Method exception-->"+e);
- e.printStackTrace();
- }
- return true;
- }
相关工程后续会上传。
1.点击下载工程1
Android(java方法)上实现mp4的分割和拼接 (二)的更多相关文章
- Android(java方法)上实现mp4的分割和拼接 (一)
最近正在处理android上的mp4切割问题.学习了很多mp4的知识,mp4文件按照编码类型,分为mpeg-4,avc这两种:这两种类型的mp4在后面的处理中会有不同的地方. 在Android系 ...
- Android Studio NDK开发-JNI调用Java方法
相对于NDK来说SDK里面有更多API可以调用,有时候我们在做NDK开发的时候,需要在JNI直接Java中的方法和变量,比如callback,系统信息等.... 如何在JNI中调用Java方法呢?就需 ...
- Unity调用Android Studio中的Java方法
1. 新建Unity项目: 2. Android Studio中新建EmptyActivity: 3. 新建安卓项目时记住最小版本号: 4. 将左侧项目文件浏览面板切换到Project项下,在本项根节 ...
- Oracle调用Java方法(上)如何使用LoadJava命令和如何将简单的Jar包封装成Oracle方法
最近在工作中遇到了遇到了一个需求需要将TIPTOP中的数据导出成XML并上传到FTP主机中,但是4GL这方面的文档比较少最终决定使用Oracle调用Java的方法,在使用的过程中发现有很多的坑,大部分 ...
- 068 01 Android 零基础入门 01 Java基础语法 08 Java方法 06 参数传递问题——基本数据类型传值
068 01 Android 零基础入门 01 Java基础语法 08 Java方法 06 参数传递问题--基本数据类型传值 本文知识点:参数传递问题--基本数据类型传值 说明:因为时间紧张,本人写博 ...
- 067 01 Android 零基础入门 01 Java基础语法 08 Java方法 05 数组作为方法参数
067 01 Android 零基础入门 01 Java基础语法 08 Java方法 05 数组作为方法参数 本文知识点:数组作为方法参数 说明:因为时间紧张,本人写博客过程中只是对知识点的关键步骤进 ...
- 066 01 Android 零基础入门 01 Java基础语法 08 Java方法 02 带参有返回值方法
066 01 Android 零基础入门 01 Java基础语法 08 Java方法 04 带参有返回值方法 本文知识点:带参有返回值方法 说明:因为时间紧张,本人写博客过程中只是对知识点的关键步骤进 ...
- Android进程so注入Hook java方法
本文博客链接:http://blog.csdn.net/qq1084283172/article/details/53769331 Andorid的Hook方式比较多,现在来学习下,基于Android ...
- maven project中,在main方法上右键Run as Java Application时,提示错误:找不到或无法加载主类XXX.XXXX.XXX
新建了一个maven project项目,经过一大堆的修改操作之后,突然发现在main方法上右键运行时,竟然提示:错误:找不到或无法加载主类xxx.xxx.xxx可能原因1.eclipse出问题了,在 ...
随机推荐
- (转)减少oracle sql回表次数 提高SQL查询性能
要写出高效的SQL,那么必须必须得清楚SQL执行路径,介绍如何提高SQL性能的文章很多,这里不再赘述,本人来谈谈如何从 减少SQL回表次数 来提高查询性能,因为回表将导致扫描更多的数据块. 我们大家都 ...
- Linux学习-CentOS 7.x 预设启动的服务简易说明
这里 仅介绍几个很常见的 daemons 而已,更多的信息呢,就得要麻烦你自己使用 systemctl list-unit-files --type=service 去查询.底下的建议主要是针对 Li ...
- IAR调试时出现IAR one or more breakpoints could not be set and have been disabled的解决办法
问题:在IAR调试时,单步执行的时候绿色箭头一直指向汇编界面,不指向C语言界面,并且不能在C语言界面设置断点,以及在代码编辑界面,设置断点,点调试时总提示IAR one or more breakpo ...
- Leetcode 456.132模式
132模式 给定一个整数序列:a1, a2, ..., an,一个132模式的子序列 ai, aj, ak 被定义为:当 i < j < k 时,ai < ak < aj.设计 ...
- Django创建并连接数据库(实现增删改查)--第二版
注意点一: url里面的地址,不只是html页面,准确说是views视图里面对应的函数方法 <!DOCTYPE html> <html lang="en"> ...
- Linux定时任务Crontab命令详解 转
linux 系统则是由 cron (crond) 这个系统服务来控制的.Linux 系统上面原本就有非常多的计划性工作,因此这个系统服务是默认启动的.另 外, 由于使用者自己也可以设置计划任务,所 ...
- SJTU Summer Camp
Day -2,-1 提前坐飞机来到了上海,在旁边的酒店住下来,晚上去了外滩,在黄浦江边吹着晚风,依旧感慨万千,在衡中高三的一年竟然已经过去,经常出现在噩梦中的高考也已成为历史,然而命运可能并未就此改变 ...
- 刷题总结——旅馆(bzoj1593线段树)
题目: Description 奶牛们最近的旅游计划,是到苏必利尔湖畔,享受那里的湖光山色,以及明媚的阳光.作为整个旅游的策划者和负责人,贝茜选择在湖边的一家著名的旅馆住宿.这个巨大的旅馆一共有N ( ...
- [暑假集训--数位dp]LightOj1205 Palindromic Numbers
A palindromic number or numeral palindrome is a 'symmetrical' number like 16461 that remains the sam ...
- Mysql常用语句记录
建表语句,带自增字段 create table test ( id int auto_increment primary key, name ) not null, password ) not nu ...