前言
下一個(gè)項(xiàng)目有一個(gè)直播的功能,所以需要提前研究一下,直接集成FFmpeg比較復(fù)雜坑也比較多(我是道聽(tīng)途說(shuō)的自己沒(méi)有嘗試??),朋友推薦了ijkplayer、七牛和vlc,于是我花了兩天時(shí)間研究了一下ijkplayer,過(guò)程非??部溃龅降膯?wèn)題不下10個(gè),中間曾一度想過(guò)放棄然后換其他的,但是看了一些文章之后還是從頭到尾做出來(lái)了,做事也算是有始有終,給自己一個(gè)贊??。其實(shí)集成的過(guò)程還是比較簡(jiǎn)單的,按照教程一步一步來(lái)就可以了,如果正常的話可能一兩個(gè)小時(shí)就搞定了,我主要是想把我遇到的問(wèn)題以及解決方法記錄一下與大家共同學(xué)習(xí)提高!
環(huán)境準(zhǔn)備
集成ijkplayer需要電腦中安裝git和yasm,然后好像還需要一個(gè)pkg-config(不確定),安裝git、yasm和pkg-config可以使用homebrew。
檢查自己電腦是不是安裝了homebrew、git、yasm可以打開(kāi)終端依次輸入:
brew -v
git --version
yasm --version
pkg-config --verison
我的已經(jīng)安裝了

安裝成功可以忽略以下內(nèi)容直接查看第一章節(jié),如果沒(méi)有安裝成功請(qǐng)繼續(xù),首先打開(kāi)終端
安裝homebrew:
/usr/bin/ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install)"
如果想卸載homebrew
/usr/bin/ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/uninstall)"
homebrew的用法:
安裝軟件,如:brew install oclint
卸載軟件,如:brew uninstall oclint
搜索軟件,如:brew search oclint
更新軟件,如:brew upgrade oclint
查看安裝列表, 如:brew list
更新Homebrew,如:brew update
使用homebrew安裝git
brew install git
使用homebrew安裝yasm
brew install yasm
使用homebrew安裝pkg-config
brew install pkg-config
一、使用git克隆倉(cāng)庫(kù)到本地
ijkplayer是bilibili的一個(gè)開(kāi)源項(xiàng)目https://github.com/Bilibili/ijkplayer
在合適的位置新建文件夾(不想新建也可以的??),我是在桌面上新建文件夾命名為showcase,然后進(jìn)入對(duì)應(yīng)文件夾
cd /Users/jizhigang/Desktop/showcase //這是我的路徑啊看準(zhǔn)嘍
clone一個(gè)倉(cāng)庫(kù)有兩種方式,使用https方式和SSH方式,我們選擇https方式,使用SSH方式繼續(xù)執(zhí)行可能出錯(cuò)??

//克隆倉(cāng)庫(kù)到指定位置并重命名為ijkplayer-ios
git clone https://github.com/Bilibili/ijkplayer.git ijkplayer-ios
有時(shí)候clone倉(cāng)庫(kù)時(shí)會(huì)出現(xiàn)問(wèn)題
error: RPC failed; curl 18 transfer closed with outstanding read data remaining
fatal: The remote end hung up unexpectedly
fatal: early EOF
fatal: index-pack failed
這個(gè)錯(cuò)誤是因?yàn)轫?xiàng)目太久,tag資源文件太大,我們clone時(shí)可以這樣
//克隆倉(cāng)庫(kù)到指定位置并重命名為ijkplayer-ios,深度為1的代碼,具體什么意思我不是特別清楚就不亂解釋了啊,免得誤導(dǎo)別人
git clone https://github.com/Bilibili/ijkplayer.git ijkplayer-ios --depth=1

然后進(jìn)入文件夾ijkplayer-ios
cd ijkplayer-ios
新建分支latest并切換到新建的分支上
git checkout -B latest k0.8.8
我們平時(shí)開(kāi)發(fā)最好不要在
master分支上直接開(kāi)發(fā),所以這里我按照教程新建了分支latest。但是我們不需要提交代碼,所以這里不新建分支也是可以的,新建與否都對(duì)后續(xù)各步驟沒(méi)有影響。
可以使用命令查看當(dāng)前分支
git branch
可以看到當(dāng)前有了兩個(gè)分支latest和master,并且當(dāng)前工作在latest分支上

配置編解碼器格式支持
默認(rèn)為最少支持, 如果足夠你使用, 可以跳過(guò)這一步. 否則可以改為以下配置:
module-default.sh 更多的編解碼器/格式
module-lite-hevc.sh 較少的編解碼器/格式(包括hevc)
module-lite.sh 較少的編解碼器/格式(默認(rèn)情況)
# 進(jìn)入 config 目錄
cd config
# 刪除當(dāng)前的 module.sh 文件
rm module.sh
# 可根據(jù)需要替換為`module-default.sh`, `module-lite-hevc.sh`, `module-lite.sh`
# 創(chuàng)建軟鏈接 module.sh 指向 module-lite-hevc.sh
ln -s module-lite-hevc.sh module.sh
cd ..
cd ios
sh compile-ffmpeg.sh clean
二、下載ffmpeg
./init-ios.sh //時(shí)間稍長(zhǎng)一些

添加 https 支持
最后會(huì)生成支持 https 的靜態(tài)文件 libcrypto.a 和 libssl.a, 如果不需要可以跳過(guò)這一步
# 獲取 openssl 并初始化
./init-ios-openssl.sh
cd ios
# 在模塊文件中添加一行配置 以啟用 openssl 組件
echo 'export COMMON_FF_CFG_FLAGS="$COMMON_FF_CFG_FLAGS --enable-openssl"' >> ../config/module.sh
./compile-ffmpeg.sh clean
然后進(jìn)入ios文件夾中
cd ios
三、 編譯ffmpeg
# 如果下一步提示錯(cuò)誤`xcrun: error: SDK "iphoneos" cannot be located`, 請(qǐng)執(zhí)行`sudo xcode-select --switch /Applications/Xcode.app/Contents/Developer/`, 再重新執(zhí)行下一步
# 編譯openssl, 如果不需要https可以跳過(guò)這一步
./compile-openssl.sh all
./compile-ffmpeg.sh clean
./compile-ffmpeg.sh all //編譯
這里有可能遇到錯(cuò)誤
xcrun -sdk iphoneos clang is unable to create an executable file.
C compiler test failed.
造成這個(gè)問(wèn)題的原因有很多,這里我是因?yàn)?code>xcode路徑問(wèn)題引起的,解決方法:
sudo /usr/bin/xcode-select -switch /Applications/Xcode.app/Contents/Developer
這里還遇到了另一個(gè)問(wèn)題
AS libavcodec/arm/aacpsdsp_neon.o
./libavutil/arm/asm.S:50:9: error: unknown directive
.arch armv7-a
^
make: *** [libavcodec/arm/aacpsdsp_neon.o] Error 1
make: *** Waiting for unfinished jobs....
最新的 Xcode 已經(jīng)弱化了對(duì) 32 位的支持, 解決方法:
在compile-ffmpeg.sh中刪除armv7, 修改如:
FF_ALL_ARCHS_IOS8_SDK="arm64 i386 x86_64"
再重新執(zhí)行出現(xiàn)錯(cuò)誤的命令:./compile-ffmpeg.sh all



然后再次執(zhí)行
./compile-ffmpeg.sh clean
./compile-ffmpeg.sh all //編譯

打開(kāi)ios/IJKMediaPlayer并運(yùn)行
添加 openssl相關(guān)包以支持https
如果不使用
https, 可以跳過(guò)此步, 直接開(kāi)始打包framwork
如果使用
https, 那么需要手動(dòng)給IJKMediaFramework添加libcrypto.a和libssl.a文件, 默認(rèn)不會(huì)添加
ps: 這兩個(gè)依賴庫(kù)的目錄為:
ijkplayer-ios/ios/build/universal/lib, 只有進(jìn)行了上面跟openssl相關(guān)的操作, 才會(huì)在這個(gè)目錄下有生成libcrypto.a和libssl.a


然后以此打開(kāi)build->universal->lib選擇libcrypto.a 和libssl.a

打開(kāi)工程可以看到

四、打包 framwork

大家會(huì)發(fā)現(xiàn)除了
IJKMediaFramework這個(gè)target, 還有一個(gè)叫IJKMediaFrameworkWithSSL, 但是不推薦使用這個(gè), 因?yàn)榇蟛糠只?ijkplayer的第三方框架都是使用的前者, 你把后者導(dǎo)入項(xiàng)目還是會(huì)報(bào)找不到包的錯(cuò)誤, 就算你要支持https也推薦使用前者, 然后按照上一步添加openssl即可支持
首先將debug改為release

分別編譯模擬器和真機(jī)


編譯的時(shí)候可能會(huì)遇到兩個(gè)問(wèn)題,按照步驟注釋掉就可以了


注釋掉這兩行代碼之后再分別選擇真機(jī)和模擬器進(jìn)行編譯command+b
成功之后可以看到

選中IJKMediaFramework.framework右鍵show in finder可以看到

合并真機(jī)和模擬器的framework,我們要合并的其實(shí)是這兩個(gè)文件

首先我們cd到products文件夾中

然后執(zhí)行: lipo -create 真機(jī)framework路徑 模擬器framework路徑 -output 合并的文件路徑
lipo -create Release-iphoneos/IJKMediaFramework.framework/IJKMediaFramework Release-iphonesimulator/IJKMediaFramework.framework/IJKMediaFramework -output IJKMediaFramework
可以看到在products文件夾中生成了一個(gè)framework文件

替換掉Release-iphoneos中的IJKMediaFramework,如圖

五、集成framwork并測(cè)試
新建工程并導(dǎo)入IJKMediaFramework.framework以及其他依賴庫(kù)

新建控制器playerViewController然后從ViewController.swift中push或者模態(tài)推出控制器進(jìn)行測(cè)試,如果直接在跟控制器ViewController.swift中測(cè)試可能會(huì)有只出聲音沒(méi)有影像的問(wèn)題
import UIKit
class playerViewController: UIViewController {
var iPlayer:IJKFFMoviePlayerController?
override func viewDidLoad() {
super.viewDidLoad()
let options:IJKFFOptions = IJKFFOptions.byDefault()
let url:URL = URL.init(string: "rtmp://live.hkstv.hk.lxdns.com/live/hks")!
self.iPlayer = IJKFFMoviePlayerController.init(contentURL: url, with: options)
var arm1 = UIViewAutoresizing.init(rawValue: 0)
arm1.insert(UIViewAutoresizing.flexibleWidth)
arm1.insert(UIViewAutoresizing.flexibleHeight)
self.iPlayer?.view.autoresizingMask = arm1
self.iPlayer?.view.backgroundColor = UIColor.white
self.iPlayer?.view.frame = CGRect.init(x: 0, y: 0, width: UIScreen.main.bounds.size.width, height: 300)
self.iPlayer?.scalingMode = .aspectFit
self.iPlayer?.shouldAutoplay = true
self.view.autoresizesSubviews = true
self.view.addSubview((self.iPlayer?.view)!)
// Do any additional setup after loading the view, typically from a nib.
}
override func didReceiveMemoryWarning() {
super.didReceiveMemoryWarning()
// Dispose of any resources that can be recreated.
}
override func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)
self.iPlayer?.prepareToPlay() //準(zhǔn)備
self.iPlayer?.play() //播放
}
override func viewWillDisappear(_ animated: Bool) {
super.viewWillDisappear(animated)
self.iPlayer?.pause()//暫停
// self.iPlayer?.shutdown() //銷毀
}
}
大功告成??

demo地址https://github.com/jzglovewjr/ijk
參考地址
homebrew的安裝和使用:
https://blog.csdn.net/sir_coding/article/details/77509602
git報(bào)錯(cuò):
https://blog.csdn.net/it_liuchengli/article/details/77040806
執(zhí)行./compile-ffmpeg.sh all報(bào)錯(cuò) :
https://github.com/Bilibili/ijkplayer/issues/1160
armv7支持問(wèn)題以及https支持問(wèn)題:
http://m.itdecent.cn/p/9743a68c2939