
概述
網(wǎng)上充斥著大量的 iOS FFmpeg 編譯的教程,有的時間比較早了,有的很多都沒有說詳細(xì),或者有個別坑作者沒有講到,有的講到到了一半,沒有例子。
所以本人參考了網(wǎng)上的編譯FFmpeg教程到集成的很多文章,然后加上本人進行了實際操作,總結(jié)出了此篇文章,希望大家如果有用到FFmpeg,以后少走一些坑。
此篇文章內(nèi)容會包含:
從新建 iOS 工程 -------> 到調(diào)用FFmpeg 命令 -------> 直到運行項目成功的本人實操的所有步驟。
目錄
- FFmpeg 簡介
- FFmpeg 的編譯
- iOS 集成 FFmpeg
- iOS 集成 FFmpeg Tool
- 優(yōu)化解決集成后問題
- iOS 調(diào)用 FFmpeg Tool
一、FFmpeg 簡介
1,F(xiàn)Fmpeg 是什么 ?
你可以把 FFmpeg 理解成一套音視頻解決方案,使用 C語言 開發(fā)的開源程序,并且免費、開源、跨平臺,它提供了錄制、轉(zhuǎn)換以及流化音視頻,編碼,特效,視音頻操作等功能,包含了非常先進的音頻/視頻編解碼庫。
(1)FFmpeg 能做什么 ?
可以實現(xiàn)播放歌曲、視頻,甚至通過命令實現(xiàn)對 視頻文件的轉(zhuǎn)碼、混合、剪輯,采集等各
種復(fù)雜處理。
(2)哪些地方用到了FFmpeg?
使 FFmpeg內(nèi)核視頻播放 : Mplayer,射手播放器 ,暴風(fēng)影音 ,KMPlayer...
使 FFmpeg作為內(nèi)核的Direct show Filter 解碼 : ffdshow,lav filters...
使 FFmpeg作為內(nèi)核的轉(zhuǎn)碼工具: ffmpeg,格式工廠...
2,F(xiàn)Fmpeg 作者

法布里斯·貝拉(FabriceBellard)是一位著名的計算機程序員,1972年生于法國Grenoble,大學(xué)就讀于巴黎高等綜合理工學(xué)院,后在國立巴黎高等電信學(xué)院攻讀。因FFmpeg、QEMU等項目而聞名業(yè)內(nèi)。他也是最快圓周率算法貝拉公式、TCCBOOT和TCC(微型C編譯器)等項目的作者。
《個人成就》:
1997年 - 他發(fā)現(xiàn)了最快速的計算圓周率的算法。
2000年 - 他化名Gérard Lantau,創(chuàng)建了 FFmpeg 項目。
2004年 - 他編寫了一個只有138KB的啟動加載程序TCCBOOT,可以在15秒內(nèi)從源代碼編譯并啟動Linux系統(tǒng)。
2009年 - 他聲稱打破了圓周率計算的世界紀(jì)錄,算出小數(shù)點后2.7萬億位,僅用一臺普通PC機。
2011年 - 他單用JavaScript寫了一個PC虛擬機Jslinux,實現(xiàn)能在瀏覽器里跑Linux 。
3,F(xiàn)Fmpeg 項目的組成
(1)基本:
- ffmpeg :是一個命令行工具,用來對視頻文件轉(zhuǎn)換格式;
- ffsever :是一個HTTP多媒體實時廣播流服務(wù)器;
- ffplay :是一個簡單的播放器,使用ffmpeg 庫解析和解碼,通過SDL顯示;
(2)其它:
- libavutil :包含一些公共的工具函數(shù);
- libavcodec :用于各種類型聲音/圖像編解碼;
- libswscale :用于視頻場景比例縮放、色彩映射轉(zhuǎn)換;
- libpostproc :用于后期效果處理;
- libavformat :用于音視頻封裝格式的生成和解析, 包括獲取解碼所需信息以生成解碼上下文結(jié)構(gòu)和讀取音視頻幀等功能。
4,F(xiàn)Fmpeg 的使用方式 (FFmpeg 代碼是包括兩部分)
(1)一部分是library
- 直接調(diào)用靜態(tài)庫,c語言的文件。
- API 都是在library ,如果直接調(diào) api 來操作視頻的話,就需要寫c或者c++了。
(2)一部分是 Tool
- 使用的是命令行,則不需要自己去編碼來實現(xiàn)視頻操作的流程,實際上tool就是把命令行轉(zhuǎn)換為api的操作,不需要使用者懂C++。
本篇文章實踐部分,主要是使用 Tool 命令方式。
二、 FFmpeg 的編譯
1. 介紹
- FFmpeg是一個多平臺多媒體處理工具,所以也可以在Mac下運行,先說一下Mac下如何安裝FFmpeg。
2. 相關(guān)地址
- FFmpeg 官網(wǎng) : http://ffmpeg.org/download.html
- FFmpeg 源碼 : https://github.com/FFmpeg/FFmpeg
- FFmpeg doc : http://www.ffmpeg.org/documentation.html
- FFmpeg wiki : https://trac.ffmpeg.org/wiki
- Homebrew 安裝網(wǎng)站 : http://brew.sh/index_zh-cn.html
- FFmpeg官方安裝教程 : https://trac.ffmpeg.org/wiki/CompilationGuide/MacOSX
3. 編譯Mac下可用 FFmpeg
- 編譯Mac下可用 FFmpeg,主要是可以在mac中,使用 FFmpeg 進行操作視頻等。
Homebrew介紹
- 簡稱brew,是Mac OSX上的軟件包管理工具,能在Mac中方便的安裝軟件或者卸載軟件。
Homebrew安裝
- 打開終端執(zhí)行
ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install)”
Homebrew命令
- 搜索軟件:brew search 軟件名, 如brew search wget
- 安裝軟件:brew install 軟件名, 如brew install wget
- 卸載軟件:brew remove 軟件名,如brew remove wget
通過Homebrew 安裝 FFmpeg
終端執(zhí)行:執(zhí)行 brew install ffmpeg --with-libvpx --with-libvorbis --with-ffplay
在終端中執(zhí)行一下命令,等待安裝完成即可:
ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install)"
安裝好Homebrew,然后終端執(zhí)行 "brew install ffmpeg",等待完成即可。
執(zhí)行結(jié)束,在終端中輸入ffmpeg,驗證是否安裝成功。
- 如果顯示大概如上圖,那么說明安裝成功
4. 編譯 iOS 下 FFmpeg
- 主要是用于iOS下進行使用FFmpeg
在進行編譯之前,我們首先需要做一些準(zhǔn)備工作安裝必備文件:
(1)安裝 gas-preprocessor
后面運行 FFmpeg-iOS-build-script 這個自動編譯腳本需要 gas-preprocessor .
安裝步驟(依次執(zhí)行下面命令):
sudo git clone https://github.com/bigsen/gas-preprocessor.git /usr/local/bin/gas
sudo cp /usr/local/bin/gas/gas-preprocessor.pl /usr/local/bin/gas-preprocessor.pl
sudo chmod 777 /usr/local/bin/gas-preprocessor.pl
sudo rm -rf /usr/local/bin/gas/
(2)安裝 yams
yasm是匯編編譯器,因為ffmpeg中為了提高效率用到了匯編指令,所以編譯時需要安裝
安裝步驟(依次執(zhí)行下面命令):
curl http://www.tortall.net/projects/yasm/releases/yasm-1.2.0.tar.gz -o yasm-1.2.0.gz
tar -zxvf yasm-1.2.0.gz
cd yasm-1.2.0
./configure && make -j 4 && sudo make install
(3)配置 FFmpeg 編譯腳本
1. 說明:
- 編譯FFmpeg可使用一個腳本:FFmpeg-iOS-build-script.sh。
- FFmpeg-iOS-build-script 是一個外國人寫的自動編譯的腳本,腳本則會自動從github中把ffmpeg源碼下到本地并開始編譯出iOS可用的庫,支持各種架構(gòu)。
手動編譯FFmpeg網(wǎng)上有一些方法,但是稍顯復(fù)雜而陳舊, 所以使用這個腳本比較方便。
2. 腳本下載地址:
git clone https://github.com/kewlbear/FFmpeg-iOS-build-script.git
3. 配置FFmpeg版本:
- 下載完成后我們可指定編譯的FFmpeg版本,修改 FF_VERSION 后面的參數(shù)就行了,本篇文章使用2.8版本。
配置裁剪(可選項):
FFmpeg庫是一個非常龐大的庫,包括編碼,解碼以及流媒體的支持等,如果不做裁剪全部編譯進來的話,最后生成的靜態(tài)庫會很大。并且有很多不需要的東西,都可以禁止掉。
1. 配置裁剪方法:
修改配置 build-ffmpeg.sh 腳本里面 CONFIGURE_FLAGS 后面的內(nèi)容即可
例如:
配置的一些參數(shù)是為了更好的選擇自己需要的功能,進行精簡和優(yōu)化ffmpeg。
我們可以手動看一下,在ffmpeg源碼目錄下,終端執(zhí)行 ./configure --help列出全部參數(shù) 。下面簡單列出部分參數(shù):
標(biāo)準(zhǔn)選項參數(shù)
--help : // 打印幫助信息 ./configure --help > ffmpegcfg.txt
--prefix=PREFIX :// 安裝程序到指定目錄[默認(rèn):空]
--bindir=DIR : // 安裝程序到指定目錄[默認(rèn):/bin]
--datadir=DIR :// 安裝數(shù)據(jù)文件到指定目錄[默認(rèn):/share/ffmpeg]
--incdir=DIR :// 安裝頭文件到指定目錄[默認(rèn):/include]
--mandir=DIR :// 安裝man page到指定路徑[默認(rèn):/share/man]
2. 配置選項參數(shù)
--disable-static :// 不構(gòu)建靜態(tài)庫[默認(rèn):關(guān)閉]
--enable-shared :// 構(gòu)建共享庫
--enable-gpl : // 允許使用GPL代碼。
--enable-nonfree :// 允許使用非免費代碼。
--disable-doc : // 不構(gòu)造文檔
--disable-avfilter :// 禁止視頻過濾器支持
--enable-small : // 啟用優(yōu)化文件尺寸大?。奚俣龋?--cross-compile : // 使用交叉編譯
--disable-hwaccels :// 禁用所有硬件加速(本機不存在硬件加速器,所有不需要)
--disable-network :// 禁用網(wǎng)絡(luò)
--disable-ffmpeg --disable-ffplay --disable-ffprobe --disable-ffserver
// 禁止ffmpeg、ffplay、ffprobe、ffserver
--disable-avdevice --disable-avcodec --disable-avcore
// 禁止libavdevice、libavcodec、libavcore
--list-decoders : // 顯示所有可用的解碼器
--list-encoders : // 顯示所有可用的編碼器
--list-hwaccels : // 顯示所有可用的硬件加速器
--list-protocols : // 顯示所有可用的協(xié)議
--list-indevs : // 顯示所有可用的輸入設(shè)備
--list-outdevs : // 顯示所有可用的輸出設(shè)備
--list-filters :// 顯示所有可用的過濾器
--list-parsers :// 顯示所有的解析器
--list-bsfs : // 顯示所有可用的數(shù)據(jù)過濾器
--disable-encoder=NAME : // 禁用XX編碼器 | disables encoder NAME
--enable-encoder=NAME : // 用XX編碼器 | enables encoder NAME
--disable-decoders : // 禁用所有解碼器 | disables all decoders
--disable-decoder=NAME : // 禁用XX解碼器 | disables decoder NAME
--enable-decoder=NAME : // 啟用XX解碼器 | enables decoder NAME
--disable-encoders : // 禁用所有編碼器 | disables all encoders
--disable-muxer=NAME : // 禁用XX混音器 | disables muxer NAME
--enable-muxer=NAME : // 啟用XX混音器 | enables muxer NAME
--disable-muxers : // 禁用所有混音器 | disables all muxers
--disable-demuxer=NAME : // 禁用XX解軌器 | disables demuxer NAME
--enable-demuxer=NAME : // 啟用XX解軌器 | enables demuxer NAME
--disable-demuxers : // 禁用所有解軌器 | disables all demuxers
--enable-parser=NAME : // 啟用XX剖析器 | enables parser NAME
--disable-parser=NAME : // 禁用XX剖析器 | disables parser NAME
--disable-parsers : // 禁用所有剖析器 | disa
運行腳本生成 FFmpeg:
配置好選項參數(shù)后就可以運行腳本,等待生成FFmpeg庫。
運行完畢就會生成:
ffmpeg-2.8、FFmpeg-iOS 、scratch、thin 等這些文件夾。
-
lib:對應(yīng)的 FFmpeg 靜態(tài)庫
-
include:對應(yīng)的 FFmpeg 頭文件
三、iOS 下 集成 FFmpeg
生成完FFmpeg庫與代碼后,我們就可以集成到iOS項目中進行使用
1,新建一個空項目,在Link Binary With Libraries 里添加
- libz.tbd
- libbz2.tbd
- libiconv.tbd
- CoreMedia.framework
- VideoToolbox.framework
- AVFoundation.framework
2,導(dǎo)入庫文件
(1)將目錄下的 include 和 lib文件夾 拖拽進項目中。
(2)設(shè)置 Header Search Paths 路徑,指向 項目中include目錄 。
例如:
$(SRCROOT)/FFmpeg_iOS/FFmpeg/include
(3)然后導(dǎo)入 #import "avformat.h" 在代碼中 寫 av_register_all() 然后進行編譯,如果沒有報錯,代表編譯成功。
四、iOS 運行 FFmpeg Tool
(1)到這一步其實已經(jīng)可以使用library庫了,如果要對音視頻進行操作,需要手動寫C++代碼去調(diào)用 API 使用FFmpeg。
(2)如果想要使用Tool工具來調(diào)用 FFmpg 的話,就是直接通過調(diào)用傳參的方式執(zhí)行ffmpeg 命令的話,就需要:
- 把剩下的 ffmpeg.h ffmpeg.c 等依賴的文件拖進項目中,并導(dǎo)入ffmpeg.h 文件。
- 然后進行調(diào)用 ffmpeg_main 函數(shù)傳遞參數(shù),執(zhí)行 ffmpeg 命令即可。
FFmpeg Tool 相關(guān)文件:
- ffmpeg.c
- ffmpeg.h
- ffmpeg_opt.c
- ffmpeg_filter.c
- cmdutils.c
- cmdutils.h
- cmdutils_common_opts.h

- config.h 文件 (在scratch目錄下四個文件都有 ):

示例 - 拖入對應(yīng)文件到工程
注釋掉無關(guān)代碼:
- 如果把相關(guān)其他文件導(dǎo)入后,編譯的時候會發(fā)現(xiàn)有一些頭文件比如 <avutil/internal.h>找不到。
不用擔(dān)心,在我了解到 應(yīng)該是 在 iOS 的 arm下不支持,也就不需要了。 - 那么這時候只需要把報紅的地方注釋掉就行了,另外如果精簡掉某些庫,那么依賴的類文件也會找不到,也是直接注釋掉相關(guān)報錯沒問題的。
頭文件
#include "compat/va_copy.h"
#include "libavresample/avresample.h"
#include "libpostproc/postprocess.h"
#include "libavutil/libm.h"
#include "libavutil/time_internal.h"
#include "libavutil/internal.h"
#include "libavutil/libm.h"
#include "libavformat/network.h"
#include "libavcodec/mathops.h"
#include "libavformat/os_support.h"
宏定義調(diào)用
FF_DISABLE_DEPRECATION_WARNINGS
函數(shù)調(diào)用
nb0_frames = nb_frames = mid_pred(ost->last_nb0_frames[0],
ost->last_nb0_frames[1],
ost->last_nb0_frames[2]);
ff_dlog(NULL, "force_key_frame: n:%f n_forced:%f prev_forced_n:%f t:%f prev_forced_t:%f -> res:%f\n",
ost->forced_keyframes_expr_const_values[FKF_N],
ost->forced_keyframes_expr_const_values[FKF_N_FORCED],
ost->forced_keyframes_expr_const_values[FKF_PREV_FORCED_N],
ost->forced_keyframes_expr_const_values[FKF_T],
ost->forced_keyframes_expr_const_values[FKF_PREV_FORCED_T],
res);
PRINT_LIB_INFO(avresample, AVRESAMPLE, flags, level);
PRINT_LIB_INFO(postproc, POSTPROC, flags, level);
其它
{ "videotoolbox_pixfmt", HAS_ARG | OPT_STRING | OPT_EXPERT, { &videotoolbox_pixfmt}, "" },
{ "videotoolbox", videotoolbox_init, HWACCEL_VIDEOTOOLBOX, AV_PIX_FMT_VIDEOTOOLBOX },
解決導(dǎo)入后編譯問題:
- GIF文件大小有限制,分開上傳了
(1)解決編譯問題1:
(2)解決編譯問題2:
如果提示以下錯誤,還需注釋以下代碼
{ "videotoolbox_pixfmt", HAS_ARG | OPT_STRING | OPT_EXPERT, { &videotoolbox_pixfmt}, "" },
{ "videotoolbox", videotoolbox_init, HWACCEL_VIDEOTOOLBOX, AV_PIX_FMT_VIDEOTOOLBOX },
(3)解決main函數(shù)重復(fù)問題
FFmpeg也有個main函數(shù),如果不改名就會沖突報錯。
- 打開 ffmpeg.c 文件,找到main函數(shù),修改為 ffmpeg_main。
- 并在 ffmpeg.h 中聲明。
五、優(yōu)化解決集成后問題
這個時候,我們應(yīng)該已經(jīng)能否編譯成功了,但是還有一些小問題,需要修改下。
1. 計數(shù)器置零問題 (ffmpeg.c的代碼中會訪問空屬性導(dǎo)致程序崩潰)
- 解決方法:
- 在 ffmpeg.c 中 找到 ffmpeg_cleanup 方法,在 term_exit() 前,將各個計數(shù)器置零:
nb_filtergraphs=0;
nb_output_files=0;
nb_output_streams=0;
nb_input_files=0;
nb_input_streams=0;
- 如下圖:
2. 命令執(zhí)行結(jié)束崩潰問題
說明
- FFmpeg 默認(rèn)執(zhí)行完會執(zhí)行 exit_program 方法結(jié)束進程,而iOS下只能啟動一個進程,如果默認(rèn)不做處理,執(zhí)行完一條命令后app就自動退出了,所以需要做一個處理。
解決方案有兩種:
(1)第一種方案(有缺點):
- 網(wǎng)上流傳的方法的方法都是找到 exit_program 函數(shù),然后注釋掉結(jié)束進程的代碼,然后調(diào)用 pthread_exit 結(jié)束線程來代替結(jié)束進程,進行解決。
這種方法的缺點:
- 執(zhí)行完 ffmpeg 的 main 函數(shù)后會回調(diào)一個code,這個回調(diào)是用于判斷命令指定過程中是否執(zhí)行錯誤的回調(diào)。但是我們?nèi)绻谕顺龅臅r候調(diào)用了pthread_exit 這樣線程就結(jié)束了,然后也不會走執(zhí)行是否成功的回調(diào)了。
- 并且這樣的話,想要監(jiān)聽到命令結(jié)束,必須要注冊一個通知,進行監(jiān)聽線程結(jié)束。
(2)第二種方案(修復(fù)缺點):
- 在命令執(zhí)行完不進行結(jié)束線程和進程,只進行 cleanup。
具體做法:
-
在 ffmpeg.c 的 ffmpeg_main 函數(shù)中,把所有調(diào)用 exit_program 函數(shù) ,改為調(diào)用 ffmpeg_cleanup 函數(shù)就可以了。
六、iOS 調(diào)用 FFmpeg Tool
目前為止,我們做完上面所有步驟后,我們已經(jīng)可以調(diào)用 FFmpeg Tool 進行各種音視頻操作了,例如 視頻合成、視頻轉(zhuǎn)Gif、視頻幀操作、視頻特效、格式轉(zhuǎn)換,視頻調(diào)速,等各種操作了。
1. FFmpeg 命令簡單介紹
使用ffmpeg命令行的大致格式如下:
ffmpeg [options] [[infile options] -i infile]... {[outfile options] outfile}...
2. FFmpeg 命令常見參數(shù)
-f 強制指定編碼格式
-i 輸出源
-t 指定輸入輸出時長
-r 指定幀率,即1S內(nèi)的幀數(shù)
-threads 指定線程數(shù)
-c:v 指定視頻的編碼格式
-ss 指定持續(xù)時長
-b:v 指定比特率
-s 指定分辨率
-y 覆蓋輸出
-filter 指定過濾器
-vf 指定視頻過濾器
-an 指定去除對音頻的影響
3. FFmpeg 命令示例 (更多使用可查看其它文檔)
格式轉(zhuǎn)換
ffmpeg -i mkv.mkv -acodec copy -vcodec copy newVideo.mp4
ffmpeg -i wav.wav -ar 44100 -y outputmusic.aac
視頻轉(zhuǎn)GIF
ffmpeg -i mkv.mkv -ss 00:00:10 -t 10 -pix_fmt rgb24 -f gif -s vga gif.gif
視頻聲音分離 與合成
ffmpeg -i mp4.mp4 -f mp3 -vn mp3.mp3
ffmpeg -i mp4.mp4 -an mp4No.mp4
ffmpeg -i mp3.mp3 -i mp4No.mp4 -map 0:0 -map 1:0 -c:v copy -c:a libfaac sound.mp4
旋轉(zhuǎn)視頻
ffmpeg -i mp4.mp4 -vf transpose=2 transpose.mp4
反轉(zhuǎn)視頻
水平翻轉(zhuǎn) :ffmpeg -i mp4.mp4 -vf hflip reversed.mp4
垂直翻轉(zhuǎn) :ffmpeg -i mp4.mp4 -vf vflip reversed.mp4
合并視頻
ffmpeg -f concat -i filelist.txt -y -acodec copy -strict -2 toName
水印字幕合成
ffmpeg -i fromName -i fromOther -filter_complex [0:v][1:v]overlay=0:H-h:enable='between(t,0,1)'[tmp];[tmp][1:v]overlay=0:H-h:enable='between(t,3,4)' -t 7 -y -strict -2 toName
4. iOS 中調(diào)用FFmpeg Tool 示例:

(1)第一種調(diào)用方式
簡單容易理解,但是麻煩。
#import "ViewController.h"
#import "ffmpeg.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
NSString *fromFile = [[NSBundle mainBundle]pathForResource:@"video.mov" ofType:nil];
NSString *toFile = @"/Users/sen/Desktop/Output/video.gif";
int argc = 4;
char **arguments = calloc(argc, sizeof(char*));
if(arguments != NULL)
{
arguments[0] = "ffmpeg";
arguments[1] = "-i";
arguments[2] = (char *)[fromFile UTF8String];
arguments[3] = (char *)[toFile UTF8String];
if (!ffmpeg_main(argc, arguments)) {
NSLog(@"生成成功");
}
}
}
@end
(2)第二種調(diào)用方式
比第一種方案,遍歷 FFmpeg 字符串命令,然后調(diào)用ffmpeg_main 傳遞參數(shù)。
/**
第二種調(diào)用方式
*/
- (void)normalRun2{
NSString *fromFile = [[NSBundle mainBundle]pathForResource:@"video.mov" ofType:nil];
NSString *toFile = @"/Users/sen/Desktop/Output/video.gif";
NSString *command_str = [NSString stringWithFormat:@"ffmpeg -i %@ %@",fromFile,toFile];
// 分割字符串
NSMutableArray *argv_array = [command_str componentsSeparatedByString:(@" ")].mutableCopy;
// 獲取參數(shù)個數(shù)
int argc = (int)argv_array.count;
// 遍歷拼接參數(shù)
char **argv = calloc(argc, sizeof(char*));
for(int i=0; i<argc; i++)
{
NSString *codeStr = argv_array[i];
argv_array[i] = codeStr;
argv[i] = (char *)[codeStr UTF8String];
}
ffmpeg_main(argc, argv);
}
結(jié)尾
- 文中Demo:https://github.com/bigsen/FFmpeg_iOS
- 到目前我們已經(jīng)可以在iOS平臺調(diào)用FFmpeg了,更多FFmpeg命令可查看其它相關(guān)文章。
- 后續(xù)的話,我會放出一款工具,可以更方便的調(diào)用FFmpeg進行處理命令。

















