接触FFmpeg有一段时间了,它是音视频开发的开源库,几乎其他所有播放器、直播平台都基于FFmpeg进行二次开发。本篇文章来总结下采用FFmpeg进行音频处理:音频混合、音频剪切、音频拼接与音频转码。

采用android studio进行开发,配置build.gradle文件:

[java] view plaincopy
  1. defaultConfig {
  2. ......
  3. externalNativeBuild {
  4. cmake {
  5. cppFlags ""
  6. }
  7. }
  8. ndk {
  9. abiFilters "armeabi-v7a"
  10. }
  11. }

另外指定cmake文件路径:

[java] view plaincopy
  1. externalNativeBuild {
  2. cmake {
  3. path "CMakeLists.txt"
  4. }
  5. }
  6. sourceSets {
  7. main {
  8. jniLibs.srcDirs = ['libs']
  9. jni.srcDirs = []
  10. }
  11. }

从FFmpeg官网下载源码,编译成ffmpeg.so动态库,并且导入相关源文件与头文件:

然后配置cMakeLists文件:

[java] view plaincopy
  1. add_library( # Sets the name of the library.
  2. audio-handle
  3. # Sets the library as a shared library.
  4. SHARED
  5. # Provides a relative path to your source file(s).
  6. src/main/cpp/ffmpeg_cmd.c
  7. src/main/cpp/cmdutils.c
  8. src/main/cpp/ffmpeg.c
  9. src/main/cpp/ffmpeg_filter.c
  10. src/main/cpp/ffmpeg_opt.c)
  11. add_library( ffmpeg
  12. SHARED
  13. IMPORTED )
  14. set_target_properties( ffmpeg
  15. PROPERTIES IMPORTED_LOCATION
  16. ../../../../libs/armeabi-v7a/libffmpeg.so )
  17. include_directories(src/main/cpp/include)
  18. find_library( log-lib
  19. log )
  20. target_link_libraries( audio-handle
  21. ffmpeg
  22. ${log-lib} )

调用FFmpeg命令行进行音频处理:

[java] view plaincopy
  1. /**
  2. * 调用ffmpeg处理音频
  3. * @param handleType handleType
  4. */
  5. private void doHandleAudio(int handleType){
  6. String[] commandLine = null;
  7. switch (handleType){
  8. case 0://转码
  9. String transformFile = PATH + File.separator + "transform.aac";
  10. commandLine = FFmpegUtil.transformAudio(srcFile, transformFile);
  11. break;
  12. case 1://剪切
  13. String cutFile = PATH + File.separator + "cut.mp3";
  14. commandLine = FFmpegUtil.cutAudio(srcFile, 10, 15, cutFile);
  15. break;
  16. case 2://合并
  17. String concatFile = PATH + File.separator + "concat.mp3";
  18. commandLine = FFmpegUtil.concatAudio(srcFile, appendFile, concatFile);
  19. break;
  20. case 3://混合
  21. String mixFile = PATH + File.separator + "mix.aac";
  22. commandLine = FFmpegUtil.mixAudio(srcFile, appendFile, mixFile);
  23. break;
  24. default:
  25. break;
  26. }
  27. executeFFmpegCmd(commandLine);
  28. }

其中,音频混音、合并、剪切和转码的FFmpeg命令行的拼接如下:

[java] view plaincopy
  1. /**
  2. * 使用ffmpeg命令行进行音频转码
  3. * @param srcFile 源文件
  4. * @param targetFile 目标文件(后缀指定转码格式)
  5. * @return 转码后的文件
  6. */
  7. public static String[] transformAudio(String srcFile, String targetFile){
  8. String transformAudioCmd = "ffmpeg -i %s %s";
  9. transformAudioCmd = String.format(transformAudioCmd, srcFile, targetFile);
  10. return transformAudioCmd.split(" ");//以空格分割为字符串数组
  11. }
  12. /**
  13. * 使用ffmpeg命令行进行音频剪切
  14. * @param srcFile 源文件
  15. * @param startTime 剪切的开始时间(单位为秒)
  16. * @param duration 剪切时长(单位为秒)
  17. * @param targetFile 目标文件
  18. * @return 剪切后的文件
  19. */
  20. @SuppressLint("DefaultLocale")
  21. public static  String[] cutAudio(String srcFile, int startTime, int duration, String targetFile){
  22. String cutAudioCmd = "ffmpeg -i %s -ss %d -t %d %s";
  23. cutAudioCmd = String.format(cutAudioCmd, srcFile, startTime, duration, targetFile);
  24. return cutAudioCmd.split(" ");//以空格分割为字符串数组
  25. }
  26. /**
  27. * 使用ffmpeg命令行进行音频合并
  28. * @param srcFile 源文件
  29. * @param appendFile 待追加的文件
  30. * @param targetFile 目标文件
  31. * @return 合并后的文件
  32. */
  33. public static  String[] concatAudio(String srcFile, String appendFile, String targetFile){
  34. String concatAudioCmd = "ffmpeg -i concat:%s|%s -acodec copy %s";
  35. concatAudioCmd = String.format(concatAudioCmd, srcFile, appendFile, targetFile);
  36. return concatAudioCmd.split(" ");//以空格分割为字符串数组
  37. }
  38. /**
  39. * 使用ffmpeg命令行进行音频混合
  40. * @param srcFile 源文件
  41. * @param mixFile 待混合文件
  42. * @param targetFile 目标文件
  43. * @return 混合后的文件
  44. */
  45. public static  String[] mixAudio(String srcFile, String mixFile, String targetFile){
  46. String mixAudioCmd = "ffmpeg -i %s -i %s -filter_complex amix=inputs=2:duration=first -strict -2 %s";
  47. mixAudioCmd = String.format(mixAudioCmd, srcFile, mixFile, targetFile);
  48. return mixAudioCmd.split(" ");//以空格分割为字符串数组
  49. }

FFmpeg处理混音的公式如下,其中sample1为源文件采样率、sample2为待混合文件采样率:

混音公式:value = sample1 + sample2 - (sample1 * sample2 / (pow(2, 16-1) - 1))

开启子线程,调用native方法进行音频处理:

[java] view plaincopy
  1. public static void execute(final String[] commands, final OnHandleListener onHandleListener){
  2. new Thread(new Runnable() {
  3. @Override
  4. public void run() {
  5. if(onHandleListener != null){
  6. onHandleListener.onBegin();
  7. }
  8. //调用ffmpeg进行处理
  9. int result = handle(commands);
  10. if(onHandleListener != null){
  11. onHandleListener.onEnd(result);
  12. }
  13. }
  14. }).start();
  15. }
  16. private native static int handle(String[] commands);

关键的native方法,是把java传入的字符串数组转成二级指针数组,然后调用FFmpeg源码中的run方法:

[cpp] view plaincopy
  1. JNIEXPORT jint JNICALL Java_com_frank_ffmpeg_FFmpegCmd_handle
  2. (JNIEnv *env, jclass obj, jobjectArray commands){
  3. int argc = (*env)->GetArrayLength(env, commands);
  4. char **argv = (char**)malloc(argc * sizeof(char*));
  5. int i;
  6. int result;
  7. for (i = 0; i < argc; i++) {
  8. jstring jstr = (jstring) (*env)->GetObjectArrayElement(env, commands, i);
  9. char* temp = (char*) (*env)->GetStringUTFChars(env, jstr, 0);
  10. argv[i] = malloc(1024);
  11. strcpy(argv[i], temp);
  12. (*env)->ReleaseStringUTFChars(env, jstr, temp);
  13. }
  14. //执行ffmpeg命令
  15. result =  run(argc, argv);
  16. //释放内存
  17. for (i = 0; i < argc; i++) {
  18. free(argv[i]);
  19. }
  20. free(argv);
  21. return result;
  22. }

关于FFmpeg的run方法的源码如下,中间有部分省略:

[cpp] view plaincopy
  1. int run(int argc, char **argv)
  2. {
  3. /****************省略********************/
  4. //注册各个模块
  5. avcodec_register_all();
  6. #if CONFIG_AVDEVICE
  7. avdevice_register_all();
  8. #endif
  9. avfilter_register_all();
  10. av_register_all();
  11. avformat_network_init();
  12. show_banner(argc, argv, options);
  13. term_init();
  14. /****************省略********************/
  15. //解析命令选项与打开输入输出文件
  16. int ret = ffmpeg_parse_options(argc, argv);
  17. if (ret < 0)
  18. exit_program(1);
  19. /****************省略********************/
  20. //文件转换
  21. if (transcode() < 0)
  22. exit_program(1);
  23. /****************省略********************/
  24. //退出程序操作:关闭文件、释放内存
  25. exit_program(received_nb_signals ? 255 : main_return_code);
  26. ffmpeg_cleanup(0);
  27. }

其中,最关键的是文件转换部分,源码如下:

[cpp] view plaincopy
  1. static int transcode(void)
  2. {
  3. int ret, i;
  4. AVFormatContext *os;
  5. OutputStream *ost;
  6. InputStream *ist;
  7. int64_t timer_start;
  8. int64_t total_packets_written = 0;
  9. //转码方法初始化
  10. ret = transcode_init();
  11. if (ret < 0)
  12. goto fail;
  13. if (stdin_interaction) {
  14. av_log(NULL, AV_LOG_INFO, "Press [q] to stop, [?] for help\n");
  15. }
  16. timer_start = av_gettime_relative();
  17. #if HAVE_PTHREADS
  18. if ((ret = init_input_threads()) < 0)
  19. goto fail;
  20. #endif
  21. //transcode循环处理
  22. while (!received_sigterm) {
  23. int64_t cur_time= av_gettime_relative();
  24. //如果遇到"q"命令,则退出循环
  25. if (stdin_interaction)
  26. if (check_keyboard_interaction(cur_time) < 0)
  27. break;
  28. //判断是否还有输出流
  29. if (!need_output()) {
  30. av_log(NULL, AV_LOG_VERBOSE, "No more output streams to write to, finishing.\n");
  31. break;
  32. }
  33. ret = transcode_step();
  34. if (ret < 0 && ret != AVERROR_EOF) {
  35. char errbuf[128];
  36. av_strerror(ret, errbuf, sizeof(errbuf));
  37. av_log(NULL, AV_LOG_ERROR, "Error while filtering: %s\n", errbuf);
  38. break;
  39. }
  40. //打印音视频流信息
  41. print_report(0, timer_start, cur_time);
  42. }
  43. #if HAVE_PTHREADS
  44. free_input_threads();
  45. #endif
  46. //文件末尾最后一个stream,刷新解码器buffer
  47. for (i = 0; i < nb_input_streams; i++) {
  48. ist = input_streams[i];
  49. if (!input_files[ist->file_index]->eof_reached && ist->decoding_needed) {
  50. process_input_packet(ist, NULL, 0);
  51. }
  52. }
  53. flush_encoders();
  54. term_exit();
  55. //写文件尾,关闭文件
  56. for (i = 0; i < nb_output_files; i++) {
  57. os = output_files[i]->ctx;
  58. if ((ret = av_write_trailer(os)) < 0) {
  59. av_log(NULL, AV_LOG_ERROR, "Error writing trailer of %s: %s", os->filename, av_err2str(ret));
  60. if (exit_on_error)
  61. exit_program(1);
  62. }
  63. }
  64. //关闭所有编码器
  65. for (i = 0; i < nb_output_streams; i++) {
  66. ost = output_streams[i];
  67. if (ost->encoding_needed) {
  68. av_freep(&ost->enc_ctx->stats_in);
  69. }
  70. total_packets_written += ost->packets_written;
  71. }
  72. if (!total_packets_written && (abort_on_flags & ABORT_ON_FLAG_EMPTY_OUTPUT)) {
  73. av_log(NULL, AV_LOG_FATAL, "Empty output\n");
  74. exit_program(1);
  75. }
  76. //关闭所有解码器
  77. for (i = 0; i < nb_input_streams; i++) {
  78. ist = input_streams[i];
  79. if (ist->decoding_needed) {
  80. avcodec_close(ist->dec_ctx);
  81. if (ist->hwaccel_uninit)
  82. ist->hwaccel_uninit(ist->dec_ctx);
  83. }
  84. }
  85. //省略最后的释放内存
  86. return ret;
  87. }

好了,使用FFmpeg进行音频剪切、混音、拼接与转码介绍完毕。如果各位有什么问题或者建议,欢迎交流。

源码:https://github.com/xufuji456/FFmpegAndroid。如果对您有帮助,麻烦fork和star。

android端采用FFmpeg进行音频混合与拼接剪切相关推荐

  1. android端采用FFmpeg进行视频剪切、转码与添加水印

    前两篇文章介绍过FFmpeg进行音频处理.音视频处理:android端采用FFmpeg进行音频混合与拼接剪切, android端采用FFmpeg进行音视频合成与分离.关于FFmpeg涉及文件导入以及c ...

  2. android端采用FFmpeg进行音视频合成与分离

    上一篇文章谈到音频剪切.混音.拼接与转码,也详细介绍cMake配置与涉及FFmpeg文件的导入: android端采用FFmpeg进行音频混合与拼接剪切 .现在接着探讨音视频的合成与分离. 1.音频提 ...

  3. android 混音 源码,FFmpegAndroid android 端基于 FFmpeg 实现音频剪切、拼接、转码、混音、编解码;视频剪切、水印、截图、转码、编 @codeKK c开源站...

    android 端基于 FFmpeg 库的使用 添加编译 ffmpeg.shine.mp3lame.x264 源码的参考脚本 目前音视频相关处理: 音频剪切.拼接 音频混音 音频转码 音视频合成 音频 ...

  4. FFmpeg音频处理——音频混合、拼接、剪切、转码

    本文是一篇关于 FFmpeg 音频处理的文章,转载详情 可见文章末尾~~ 接触FFmpeg有一段时间了,它是音视频开发的开源库,几乎其他所有播放器.直播平台都基于FFmpeg进行二次开发. 本篇文章来 ...

  5. Android端使用FFmpeg进行视频画面拼接

    今天看到抖音上,有人发布水平拼接的短视频.这创意不错,本身是两个不同的短视频,通过视频拼接技术把两短视频拼在一起.除了视频进行水平拼接,音频也混合在一起.前期还需要对那两视频进行预处理:包括时长.分辨 ...

  6. 用ffmpeg进行音频格式转换、剪切、合并、音量调整等

    为什么网上一大堆类似文章,我还要再发一篇?主要是,ffmpeg是分版本的,在一个版本上能用的,在另外一个版本上,未必能用. 这里发布的命令行,在ffmpeg version 5.1-full_buil ...

  7. FFmpeg手撕视频(Android端)

    前言 FFmpeg是非常强大的音视频处理工具,我们可以使用它来处理视频合成.剪辑.加特效等等操作. 官方文档至上 FFmpeg的官方文档 FFmpeg的官方文档命令真的是太多太多,而且都是英文,感觉精 ...

  8. Android端+java后端+servlet+MySQL的型男塑造平台【前后端源代码+论文+答辩ppt】

    活动地址:毕业季·进击的技术er 目录 前言 第一章 绪论 1.1 背景和意义 1.2 国内外研究现状 1.3 论文研究目标与内容 1.4.减肥瘦身相关概念与计算方式介绍 第二章 需求分析 2.1 平 ...

  9. 【Flutter】Flutter 混合开发 ( Flutter 与 Native 通信 | Android 端实现 MethodChannel 通信 )

    文章目录 前言 一.Android 端 MethodChannel 构造函数 二.Android 端 setMethodCallHandler 方法 三.Android 端实现 MethodChann ...

最新文章

  1. React 项目--button 绑定事件(15)
  2. MySQL数据库查询中的特殊命令
  3. PPT | 云客堂——云服务助力Java 应用程序开发及部署
  4. 起名算法 php,PHP实现各种经典算法详解
  5. Akamai托管服务应对多重网络性能挑战
  6. python 学习(1)
  7. HDU 5730——Shell Necklace
  8. List、Map、Set 三个接口,存取元素时的特点
  9. mysql定义变量字符串类型_mysqli_stmt :: bind_param():类型定义字符串中的元素数量与绑定变量的数量不匹配...
  10. 如何查看Tomcat版本信息
  11. SpringBoot中的SearchStrategy介绍
  12. 白泽六足机器人_ros_v1——零件准备
  13. Elasticsearch系列—倒排索引原理
  14. xf0x9f java mysql_MySql导入数据报错Incorrect string value: ‘xF0x9Fx90x82’
  15. Win10 添加虚拟打印机
  16. HDU1788 【中国剩余定理】
  17. clang diagnostic用例
  18. 基于OpenXR,Collabora推开源VI-SLAM AR/VR定位系统
  19. java面试常问问题及答案,附源代码
  20. 骗子预警,大家注意,飞鸟资源网 www.fn121.c*o*m

热门文章

  1. 包管理工具,npm、yarn、pnpm
  2. acr122_ACR的完整形式是什么?
  3. 备战秋招面试350道面试大全
  4. 如何生成和使用CLIPS动态链接库
  5. SCOUT 薄膜分析软件
  6. SVM对于高斯核函数的理解
  7. 能力素质有所欠缺_心理素质要怎么锻炼?
  8. 未来应用陈鸿:被微信封掉公众号后怎么办?给微信创业者的10点真诚建议
  9. 小米手机防盗nbsp;方法、安全及可…
  10. xp无法访问win7计算机提示无权限,ghost xp访问win7共享无权限怎么解决