亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Android > 正文

android采用FFmpeg實現音頻混合與拼接剪切

2019-10-21 21:32:59
字體:
來源:轉載
供稿:網友

接觸FFmpeg有一段時間了,它是音視頻開發的開源庫,幾乎其他所有播放器、直播平臺都基于FFmpeg進行二次開發。本篇文章來總結下采用FFmpeg進行音頻處理:音頻混合、音頻剪切、音頻拼接與音頻轉碼。

采用android studio進行開發,配置build.gradle文件:

defaultConfig { ...... externalNativeBuild {  cmake {  cppFlags ""  } } ndk {  abiFilters "armeabi-v7a" } }

另外指定cmake文件路徑:

externalNativeBuild { cmake {  path "CMakeLists.txt" } } sourceSets { main {  jniLibs.srcDirs = ['libs']  jni.srcDirs = [] } }

從FFmpeg官網下載源碼,編譯成ffmpeg.so動態庫,并且導入相關源文件與頭文件:

android,FFmpeg,音頻混合,拼接剪切

然后配置cMakeLists文件:

add_library( # Sets the name of the library.  audio-handle   # Sets the library as a shared library.  SHARED   # Provides a relative path to your source file(s).  src/main/cpp/ffmpeg_cmd.c  src/main/cpp/cmdutils.c  src/main/cpp/ffmpeg.c  src/main/cpp/ffmpeg_filter.c  src/main/cpp/ffmpeg_opt.c)add_library( ffmpeg  SHARED  IMPORTED )set_target_properties( ffmpeg   PROPERTIES IMPORTED_LOCATION   ../../../../libs/armeabi-v7a/libffmpeg.so )include_directories(src/main/cpp/include)find_library( log-lib  log )target_link_libraries( audio-handle   ffmpeg   ${log-lib} )調用FFmpeg命令行進行音頻處理: /** * 調用ffmpeg處理音頻 * @param handleType handleType */ private void doHandleAudio(int handleType){ String[] commandLine = null; switch (handleType){  case 0://轉碼  String transformFile = PATH + File.separator + "transform.aac";  commandLine = FFmpegUtil.transformAudio(srcFile, transformFile);  break;  case 1://剪切  String cutFile = PATH + File.separator + "cut.mp3";  commandLine = FFmpegUtil.cutAudio(srcFile, 10, 15, cutFile);  break;  case 2://合并  String concatFile = PATH + File.separator + "concat.mp3";  commandLine = FFmpegUtil.concatAudio(srcFile, appendFile, concatFile);  break;  case 3://混合  String mixFile = PATH + File.separator + "mix.aac";  commandLine = FFmpegUtil.mixAudio(srcFile, appendFile, mixFile);  break;  default:  break; } executeFFmpegCmd(commandLine); }

其中,音頻混音、合并、剪切和轉碼的FFmpeg命令行的拼接如下:

/** * 使用ffmpeg命令行進行音頻轉碼 * @param srcFile 源文件 * @param targetFile 目標文件(后綴指定轉碼格式) * @return 轉碼后的文件 */ public static String[] transformAudio(String srcFile, String targetFile){ String transformAudioCmd = "ffmpeg -i %s %s"; transformAudioCmd = String.format(transformAudioCmd, srcFile, targetFile); return transformAudioCmd.split(" ");//以空格分割為字符串數組 }  /** * 使用ffmpeg命令行進行音頻剪切 * @param srcFile 源文件 * @param startTime 剪切的開始時間(單位為秒) * @param duration 剪切時長(單位為秒) * @param targetFile 目標文件 * @return 剪切后的文件 */ @SuppressLint("DefaultLocale") public static String[] cutAudio(String srcFile, int startTime, int duration, String targetFile){ String cutAudioCmd = "ffmpeg -i %s -ss %d -t %d %s"; cutAudioCmd = String.format(cutAudioCmd, srcFile, startTime, duration, targetFile); return cutAudioCmd.split(" ");//以空格分割為字符串數組 }  /** * 使用ffmpeg命令行進行音頻合并 * @param srcFile 源文件 * @param appendFile 待追加的文件 * @param targetFile 目標文件 * @return 合并后的文件 */ public static String[] concatAudio(String srcFile, String appendFile, String targetFile){ String concatAudioCmd = "ffmpeg -i concat:%s|%s -acodec copy %s"; concatAudioCmd = String.format(concatAudioCmd, srcFile, appendFile, targetFile); return concatAudioCmd.split(" ");//以空格分割為字符串數組 }  /** * 使用ffmpeg命令行進行音頻混合 * @param srcFile 源文件 * @param mixFile 待混合文件 * @param targetFile 目標文件 * @return 混合后的文件 */ public static String[] mixAudio(String srcFile, String mixFile, String targetFile){ String mixAudioCmd = "ffmpeg -i %s -i %s -filter_complex amix=inputs=2:duration=first -strict -2 %s"; mixAudioCmd = String.format(mixAudioCmd, srcFile, mixFile, targetFile); return mixAudioCmd.split(" ");//以空格分割為字符串數組 }

FFmpeg處理混音的公式如下,其中sample1為源文件采樣率、sample2為待混合文件采樣率:

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

開啟子線程,調用native方法進行音頻處理:

public static void execute(final String[] commands, final OnHandleListener onHandleListener){ new Thread(new Runnable() {  @Override  public void run() {  if(onHandleListener != null){   onHandleListener.onBegin();  }  //調用ffmpeg進行處理  int result = handle(commands);  if(onHandleListener != null){   onHandleListener.onEnd(result);  }  } }).start(); } private native static int handle(String[] commands);

關鍵的native方法,是把java傳入的字符串數組轉成二級指針數組,然后調用FFmpeg源碼中的run方法:

JNIEXPORT jint JNICALL Java_com_frank_ffmpeg_FFmpegCmd_handle(JNIEnv *env, jclass obj, jobjectArray commands){ int argc = (*env)->GetArrayLength(env, commands); char **argv = (char**)malloc(argc * sizeof(char*)); int i; int result; for (i = 0; i < argc; i++) { jstring jstr = (jstring) (*env)->GetObjectArrayElement(env, commands, i); char* temp = (char*) (*env)->GetStringUTFChars(env, jstr, 0); argv[i] = malloc(1024); strcpy(argv[i], temp); (*env)->ReleaseStringUTFChars(env, jstr, temp); } //執行ffmpeg命令 result = run(argc, argv); //釋放內存 for (i = 0; i < argc; i++) { free(argv[i]); } free(argv); return result;}

關于FFmpeg的run方法的源碼如下,中間有部分省略:

int run(int argc, char **argv){ /****************省略********************/ //注冊各個模塊 avcodec_register_all();#if CONFIG_AVDEVICE avdevice_register_all();#endif avfilter_register_all(); av_register_all(); avformat_network_init(); show_banner(argc, argv, options); term_init(); /****************省略********************/ //解析命令選項與打開輸入輸出文件 int ret = ffmpeg_parse_options(argc, argv); if (ret < 0) exit_program(1); /****************省略********************/ //文件轉換 if (transcode() < 0) exit_program(1); /****************省略********************/ //退出程序操作:關閉文件、釋放內存 exit_program(received_nb_signals ? 255 : main_return_code); ffmpeg_cleanup(0);}

其中,最關鍵的是文件轉換部分,源碼如下:

static int transcode(void){ int ret, i; AVFormatContext *os; OutputStream *ost; InputStream *ist; int64_t timer_start; int64_t total_packets_written = 0; //轉碼方法初始化 ret = transcode_init(); if (ret < 0) goto fail;  if (stdin_interaction) { av_log(NULL, AV_LOG_INFO, "Press [q] to stop, [?] for help/n"); } timer_start = av_gettime_relative(); #if HAVE_PTHREADS if ((ret = init_input_threads()) < 0) goto fail;#endif //transcode循環處理 while (!received_sigterm) { int64_t cur_time= av_gettime_relative();  //如果遇到"q"命令,則退出循環 if (stdin_interaction)  if (check_keyboard_interaction(cur_time) < 0)  break;  //判斷是否還有輸出流 if (!need_output()) {  av_log(NULL, AV_LOG_VERBOSE, "No more output streams to write to, finishing./n");  break; }  ret = transcode_step(); if (ret < 0 && ret != AVERROR_EOF) {  char errbuf[128];  av_strerror(ret, errbuf, sizeof(errbuf));   av_log(NULL, AV_LOG_ERROR, "Error while filtering: %s/n", errbuf);  break; }  //打印音視頻流信息 print_report(0, timer_start, cur_time); }#if HAVE_PTHREADS free_input_threads();#endif  //文件末尾最后一個stream,刷新解碼器buffer for (i = 0; i < nb_input_streams; i++) { ist = input_streams[i]; if (!input_files[ist->file_index]->eof_reached && ist->decoding_needed) {  process_input_packet(ist, NULL, 0); } } flush_encoders(); term_exit();  //寫文件尾,關閉文件 for (i = 0; i < nb_output_files; i++) { os = output_files[i]->ctx; if ((ret = av_write_trailer(os)) < 0) {  av_log(NULL, AV_LOG_ERROR, "Error writing trailer of %s: %s", os->filename, av_err2str(ret));  if (exit_on_error)  exit_program(1); } }  //關閉所有編碼器 for (i = 0; i < nb_output_streams; i++) { ost = output_streams[i]; if (ost->encoding_needed) {  av_freep(&ost->enc_ctx->stats_in); } total_packets_written += ost->packets_written; }  if (!total_packets_written && (abort_on_flags & ABORT_ON_FLAG_EMPTY_OUTPUT)) { av_log(NULL, AV_LOG_FATAL, "Empty output/n"); exit_program(1); }  //關閉所有解碼器 for (i = 0; i < nb_input_streams; i++) { ist = input_streams[i]; if (ist->decoding_needed) {  avcodec_close(ist->dec_ctx);  if (ist->hwaccel_uninit)  ist->hwaccel_uninit(ist->dec_ctx); } }  //省略最后的釋放內存 return ret;}

好了,使用FFmpeg進行音頻剪切、混音、拼接與轉碼介紹完畢。如果各位有什么問題或者建議,歡迎交流。

源碼:鏈接地址。如果對您有幫助,麻煩fork和star。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。


注:相關教程知識閱讀請移步到Android開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲最大福利网| 欧美整片在线观看| 精品久久久久国产| 日韩国产在线播放| 激情成人中文字幕| 亚洲精品国产精品自产a区红杏吧| 综合av色偷偷网| 国产精品亚洲自拍| 美女国内精品自产拍在线播放| 91亚洲一区精品| 最近2019中文字幕第三页视频| 国产精品麻豆va在线播放| 欧美韩日一区二区| 国产精品久久激情| 美女av一区二区| 日本精品在线视频| 日韩欧美国产黄色| 国产成人福利视频| 亚洲国产高清高潮精品美女| 亚洲欧美综合v| 丝袜一区二区三区| 日韩免费在线电影| 色综合91久久精品中文字幕| 不卡av在线播放| 日韩av在线免费| 欧美专区日韩视频| 欧美性开放视频| 欧美性高跟鞋xxxxhd| 国产精品私拍pans大尺度在线| 久久不射电影网| 国产精品伦子伦免费视频| 日韩日本欧美亚洲| 日本在线观看天堂男亚洲| 国产精品久久久久久久一区探花| 日韩综合中文字幕| 一区二区三区www| 久久久久久久久久久久av| 精品久久久久久久久久ntr影视| 久久精品欧美视频| 日韩一区二区欧美| 亚洲精品短视频| 欧美成人精品影院| 91精品久久久久久久久青青| 在线观看日韩视频| 亚洲精品中文字幕有码专区| 欧美视频免费在线观看| 尤物九九久久国产精品的特点| 亚洲第五色综合网| 亚洲日本中文字幕| 欧美一区二区三区……| 久久精品99久久久香蕉| 色综合亚洲精品激情狠狠| 久久国产色av| 欧美第一黄色网| 欧美第一淫aaasss性| 亚洲欧美日韩天堂| 91精品久久久久久久久中文字幕| 亚洲一区二区在线| 久久人人爽亚洲精品天堂| 人人爽久久涩噜噜噜网站| 久久精品国产亚洲精品| 成人性生交xxxxx网站| 少妇高潮 亚洲精品| 97超碰国产精品女人人人爽| 亚洲美女免费精品视频在线观看| 日韩av网站在线| 麻豆一区二区在线观看| 日韩在线免费av| 7777免费精品视频| 欧美性色视频在线| 国产成人在线一区| 国产极品jizzhd欧美| 国产成人jvid在线播放| 国产精品第一第二| 久久久免费av| 国产91色在线播放| 日韩成人在线视频观看| 欧美性生交xxxxxdddd| 国产精品一二三视频| 日韩中文在线中文网在线观看| 亚洲欧美日韩高清| 色樱桃影院亚洲精品影院| 久久九九国产精品怡红院| 欧美自拍视频在线| 日韩欧美中文第一页| 日韩亚洲精品视频| 在线观看欧美成人| 91精品综合视频| 92福利视频午夜1000合集在线观看| 亚洲精品午夜精品| 国产精品观看在线亚洲人成网| 中文字幕亚洲色图| 日本高清不卡的在线| 久热精品视频在线观看一区| 97在线观看视频国产| 亚洲第一二三四五区| 成人精品一区二区三区| 亚洲色图18p| 色婷婷综合成人| 精品久久久久久国产91| 欧美日韩一区二区在线播放| 亚洲男人7777| 欧美日韩xxxxx| 国产精品91免费在线| 69国产精品成人在线播放| 精品国产依人香蕉在线精品| 日韩av在线网站| 岛国av一区二区在线在线观看| 日韩欧美极品在线观看| 国产日韩中文字幕在线| 欧美一级视频在线观看| 亚洲一区二区中文字幕| 日韩精品中文在线观看| 欧美一级高清免费| 免费91在线视频| 中文字幕日韩欧美在线| 亚洲精品一区久久久久久| 欧美性猛交xxxx免费看| 91国产中文字幕| 欧美极品xxxx| 亚洲天堂男人天堂| 国产欧美欧洲在线观看| 亚洲男人av电影| 国产精品激情av电影在线观看| 亚洲成**性毛茸茸| 丝袜情趣国产精品| 国产精品第三页| 亚洲老头老太hd| 欧美一级电影在线| 欧美午夜性色大片在线观看| 尤物yw午夜国产精品视频明星| 久久99精品国产99久久6尤物| 久久久国产一区二区三区| xxxxx91麻豆| 欧美激情国产精品| 久久不射电影网| 91欧美精品成人综合在线观看| 欧洲亚洲在线视频| 最近的2019中文字幕免费一页| 亚洲午夜精品久久久久久久久久久久| 国产精品高潮呻吟久久av黑人| 色老头一区二区三区在线观看| 91日本视频在线| 成人网在线观看| 亚洲网站在线播放| 中日韩美女免费视频网址在线观看| 久久精品人人做人人爽| 日韩美女中文字幕| 日韩欧美在线播放| 国产亚洲精品久久久优势| 色爱av美腿丝袜综合粉嫩av| 欧美激情欧美激情| 亚洲成人网久久久| 91香蕉国产在线观看| 精品中文字幕在线观看| 欧美裸身视频免费观看| 欧美激情视频免费观看| 91国在线精品国内播放| 国产精品aaa| 亚洲高清在线观看| 亚洲精品福利免费在线观看| 欧美日韩国产一区在线| 欧美亚洲国产另类| 亚洲女人天堂网|