Android音视频处理之基于MediaCodec合并音视频
Android提供了一个MediaExtractor类,可以用来分离容器中的视频track和音频track,下面的例子展示了使用MediaExtractor和MediaMuxer来实现视频的换音:
- private void muxingAudioAndVideo() throws IOException {
- MediaMuxer mMediaMuxer = new MediaMuxer(mOutputVideoPath,
- MediaMuxer.OutputFormat.MUXER_OUTPUT_MPEG_4);
- // 视频的MediaExtractor
- MediaExtractor mVideoExtractor = new MediaExtractor();
- mVideoExtractor.setDataSource(mVideoPath);
- int videoTrackIndex = -1;
- for (int i = 0; i < mVideoExtractor.getTrackCount(); i++) {
- MediaFormat format = mVideoExtractor.getTrackFormat(i);
- if (format.getString(MediaFormat.KEY_MIME).startsWith("video/")) {
- mVideoExtractor.selectTrack(i);
- videoTrackIndex = mMediaMuxer.addTrack(format);
- break;
- }
- }
- // 音频的MediaExtractor
- MediaExtractor mAudioExtractor = new MediaExtractor();
- mAudioExtractor.setDataSource(mAudioPath);
- int audioTrackIndex = -1;
- for (int i = 0; i < mAudioExtractor.getTrackCount(); i++) {
- MediaFormat format = mAudioExtractor.getTrackFormat(i);
- if (format.getString(MediaFormat.KEY_MIME).startsWith("audio/")) {
- mAudioExtractor.selectTrack(i);
- audioTrackIndex = mMediaMuxer.addTrack(format);
- }
- }
- // 添加完所有轨道后start
- mMediaMuxer.start();
- // 封装视频track
- if (-1 != videoTrackIndex) {
- MediaCodec.BufferInfo info = new MediaCodec.BufferInfo();
- info.presentationTimeUs = 0;
- ByteBuffer buffer = ByteBuffer.allocate(100 * 1024);
- while (true) {
- int sampleSize = mVideoExtractor.readSampleData(buffer, 0);
- if (sampleSize < 0) {
- break;
- }
- info.offset = 0;
- info.size = sampleSize;
- info.flags = MediaCodec.BUFFER_FLAG_SYNC_FRAME;
- info.presentationTimeUs = mVideoExtractor.getSampleTime();
- mMediaMuxer.writeSampleData(videoTrackIndex, buffer, info);
- mVideoExtractor.advance();
- }
- }
- // 封装音频track
- if (-1 != audioTrackIndex) {
- MediaCodec.BufferInfo info = new MediaCodec.BufferInfo();
- info.presentationTimeUs = 0;
- ByteBuffer buffer = ByteBuffer.allocate(100 * 1024);
- while (true) {
- int sampleSize = mAudioExtractor.readSampleData(buffer, 0);
- if (sampleSize < 0) {
- break;
- }
- info.offset = 0;
- info.size = sampleSize;
- info.flags = MediaCodec.BUFFER_FLAG_SYNC_FRAME;
- info.presentationTimeUs = mAudioExtractor.getSampleTime();
- mMediaMuxer.writeSampleData(audioTrackIndex, buffer, info);
- mAudioExtractor.advance();
- }
- }
- // 释放MediaExtractor
- mVideoExtractor.release();
- mAudioExtractor.release();
- // 释放MediaMuxer
- mMediaMuxer.stop();
- mMediaMuxer.release();
- }
MediaExtractor的接口比较简单,首先通过setDataSource()设置数据源,数据源可以是本地文件地址,也可以是网络地址:
- MediaExtractor mVideoExtractor = new MediaExtractor();
- mVideoExtractor.setDataSource(mVideoPath);
然后可以通过getTrackFormat(int index)来获取各个track的MediaFormat,通过MediaFormat来获取track的详细信息,如:MimeType、分辨率、采样频率、帧率等等:
- for (int i = 0; i < mVideoExtractor.getTrackCount(); i++) {
- MediaFormat format = mVideoExtractor.getTrackFormat(i);
- }
获取到track的详细信息后,通过selectTrack(int index)选择指定的通道:
- if (format.getString(MediaFormat.KEY_MIME).startsWith("video/")) {
- mVideoExtractor.selectTrack(i);
- break;
- }
指定通道之后就可以从MediaExtractor中读取数据了:
- while (true) {
- int sampleSize = mVideoExtractor.readSampleData(buffer, 0);
- if (sampleSize < 0) {
- break;
- }
- // do something
- mVideoExtractor.advance(); // 移动到下一帧
- }
- mVideoExtractor.release();
Android音视频处理之基于MediaCodec合并音视频的更多相关文章
- Android多媒体框架总结(1) - 利用MediaMuxer合成音视频数据流程分析
场景介绍: 设备端通过服务器传向客户端(Android手机)实时发送视频数据(H.264)和音频数据(g711a或g711u), 需要在客户端将音视频数据保存为MP4文件存放在本地,用户可以通过APP ...
- 10分钟快速上车短视频风口:基于uniapp框架创建自己的仿抖音短视APP
在今年也就是第48次发布的<中国互联网络发展状况统计报告>有这样一个数据,21年的上半年以来,我国我国网民规模达10.11亿,其中短视频用户达8.88亿.碎片化的生活场景下,短视频成为人们 ...
- iOS直播-播放基于RTMP协议的视频
iOS直播-播放基于RTMP协议的视频 流媒体协议介绍 1. 伪流媒体: 渐进式下载 : 边下边存, 文件会保存 使用http协议,也能够实现视频播放, 也能快进快退等, 体验上跟流媒体很像. 优酷, ...
- 基于语音识别、音文同步、图像OCR的字幕解决方案HtwMedia介绍
背景介绍 俗话说,“好记性不如乱笔头”,这充分说明了文字归档的重要性.如今随着微信.抖音等移动端app的使用越来越广,人们生产音.视频内容也越来越便捷.而相比语音和视频而言,文字具有易存档.易检索.易 ...
- PyQt+moviepy音视频剪辑实战2:一个剪裁视频文件精华内容留存工具的实现
专栏:Python基础教程目录 专栏:使用PyQt开发图形界面Python应用 PyQt+moviepy音视频剪辑实战 专栏:PyQt入门学习 老猿Python博文目录 老猿学5G博文目录 一.引言 ...
- android的四层体系结构,基于mvc三层结构浅析
从多方面理解Android体系结构 1.以分层的方式来看Android 安卓体系结构分为四层. 首先看一下官方关于Android体系结构的图: 1).Linux Kernel:负责硬件的驱动程序.网络 ...
- 基于RTP的H264视频数据打包解包类
from:http://blog.csdn.net/dengzikun/article/details/5807694 最近考虑使用RTP替换原有的高清视频传输协议,遂上网查找有关H264视频RTP打 ...
- javaCV开发详解之2:推流器实现,推本地摄像头视频到流媒体服务器以及摄像头录制视频功能实现(基于javaCV-FFMPEG、javaCV-openCV)
javaCV系列文章: javacv开发详解之1:调用本机摄像头视频 javaCV开发详解之2:推流器实现,推本地摄像头视频到流媒体服务器以及摄像头录制视频功能实现(基于javaCV-FFMPEG.j ...
- Android开发学习之路--基于vitamio的视频播放器(二)
终于把该忙的事情都忙得差不多了,接下来又可以开始good good study,day day up了.在Android开发学习之路–基于vitamio的视频播放器(一)中,主要讲了播放器的界面的 ...
随机推荐
- oracle常用密令大全
1.create user username identified by password;//建用户名和密码oracle ,oracle 2.grant connect,resource,dba t ...
- js运用sort对json 数组进行排序
Array.sort()方法是用来对数组项进行排序的 ,默认情况下是进行升序排列.sort() 方法可以接受一个 方法为参数. sort()排序时每次比较两个数组项都回执行这个参数,并把两个比较的数组 ...
- Mac Go 环境变量配置
GOPATH 是工作目录,就是你打代码,代码的存放目录 GOROOT 是Go的安装目录,我下载的是免安装版的 现在的Go环境变量就是设置成这个样子, 终于Bee不会报错了!!!
- ImportError: dynamic module does not define init function (initcaffe)
https://github.com/BVLC/caffe/issues/2770 $ python2 -c "import caffe" Traceback (most rece ...
- slf4j NoSuchMethodError 错误 ---- 版本冲突
java.lang.NoSuchMethodError: org.slf4j.spi.LocationAwareLogger.log(Lorg/slf4j/Marker;Ljava/lang/Stri ...
- 实验吧Web-中-登陆一下好吗??
题目上说:不要怀疑,我已经过滤了一切,还再逼你注入,哈哈哈哈哈! 可以试试,只要是输入的关键字都被过滤了,双写也被过滤掉了. 用万能密码发现,or被过滤掉了. 这里用到的是admin为:'=',密码为 ...
- [XSS防御]HttpOnly之四两拨千斤
今天看了<白帽子讲web安全>一书,顺便记录一下,HttpOnly的设置 httponly的设置值为 TRUE 时,使得Javascript无法获取到该值,有效地防御了XSS打管理员的 c ...
- 使用NtQueryInformationFile函数获得不到完整路径
#include <windows.h> #include <iostream> using namespace std; typedef struct _OBJECT_NAM ...
- 恒生UFX交易接口基本介绍
https://zhidao.baidu.com/question/203296047903136445.html 1.恒生UFT和UFX有什么区别? UFT是一个极速交易系统,UFX是一个统一接入系 ...
- Python的安装部署
Python的安装部署 参考:https://www.runoob.com/w3cnote/pycharm-windows-install.html 参考:https://jingyan.baidu. ...