Android視頻開發進階(part2-MP4文件的解析)

上一期文章我分享了一些視頻播放裏面的術語和基本概念。這一篇文章我會主要介紹容器(container format file)格式文件的細節,以最多見的MP4文件入手。而後會簡短的介紹一個標準的播放器的啓動,解析,播放流程。本篇仍是以基礎知識爲主,雖然很枯燥,可是對視頻開發的學習有很是大的好處,我本身我的的感覺就是,若是在不少專有名字,概念都不熟悉的狀況下,想要去閱讀播放器源碼會是至關困難的事情。好比Exoplayer,谷歌的分包策略就是根據播放器的組件來分包。若是不熟悉播放器的基礎構建的話,連哪一個部分的代碼在哪一個包都不知道。但願你們若是真的想進階的話仍是耐心的理解好每一個基礎概念。html

  1. Mp4格式文件的構成
  2. Mp4頭文件的構成
  3. 標準播放器的啓動流程
  4. 在線視頻播放的技術基礎(online video streaming)

1. Mp4格式文件的構成

在上期咱們大概介紹了Mp4文件的結構api

可是這樣抽象的介紹可能仍是比較難理解,咱們深刻一些。緩存

1.1 MP4究竟是個啥?

通俗的說,MP4實際上是一種格式的規範,這個規範是被ISO機構認證的,也就是說,只要你經過Codec生成了一個mp4文件,那麼這個文件的格式必須是按照ISO機構的規矩來。。。。既然是規範,那麼咱們看看到底ISO對mp4作了什麼規範:app

請你們打開連接->ISO的mp4文件規範ide

你們可能會有點懵逼,看不懂。其實這個規範很好理解,它定義了一個MP4文件裏面,哪些數據應該放在什麼位置(以字節爲單位),哪些數據的長度是多少。我截取了一段:post

你們看,上面這一段規範定義了ftyp這個頭文件header所在的位置和長度(以字節爲單位)。 至於這些頭文件是有什麼用,我在上一篇文章大概提到過,他們屬於meta data的一部分。在本章我會更詳細的介紹。學習

因此說,任何容器,包括mp4都是相似的結構化文件,只不過不一樣的格式文件ISO對其有嚴格的要求,數據的擺放順序,排列等等不一樣而已。有興趣的同窗能夠對比一下rmvb,mp4,mkv這些格式的要求有什麼不一樣,優劣勢各是什麼。ui

2.Mp4頭文件的構成

關於mp4文件的頭文件格式(meta data),蘋果官網對其進行了詳細的描述(這個介紹是基於QuickTime播放器支持的mp4文件來介紹的,quciktime播放器對mp4的要求有些許不一樣,可是差異不大,咱們能夠忽略):atom

Movie Atom翻譯

咱們不追究太多細節,有興趣的同窗能夠本身查看,咱們專一於一些基礎的信息。

首先,在Meta Data裏面,每個Header,頭文件,咱們都叫他們Atom Header(不知道咋翻譯)。Atom Header分爲Leaf Atom 和 Container Atom。前者表明一個鏈接着字符串信息的頭文件,後者是一個包含了若干個子Atom的頭文件,他們互相之間是有層級關係的(參考上圖)。每次播放器獲取了movie atom以後(moov),會根據層級關係,向下,或者向下讀取相關的其餘信息。每個頭文件都會對它的子頭文件保存位置的引用,因此只要根據mp4文件的規範獲取了最頂級的頭文件moov,就能夠順勢往下讀取其餘頭文件了。

咱們來看看mp4的頭文件結構

看起來很複雜,可是對於一個播放器來講,不少信息都不是必須。咱們須要知道的最重要的信息是採樣索引表(Sample Table Atoms).對應圖中「**stbl **」這個atom header。這個索引表保存了mp4文件全部的採樣(sample)與視頻時間的對應關係(通常以微秒爲單位),還有包括每一個採樣的大小,在mp4文件中的起始位置(以本身爲單位)。

3.標準播放器的啓動流程

那麼既然咱們已經知道一個容器文件的格式規範了,播放器就能夠經過解析容器的頭文件來控制播放(playback)了。

3.1 播放器

一般播放器由三個部分構成

  1. 讀取器(Extractor)
  2. 渲染器(TrackRenderer)
  3. 加載控制器(Load Controller)
  4. 數據源(Source)

讀取器負責從source文件讀取數據,加載控制器負責控制讀取數據的策略(好比說在線視頻播放的時候緩衝策略),渲染器負責接收讀取器讀取的數據,並渲染到屏幕上。

3.2 播放器的播放過程

在播放器能夠把數據提交給渲染器以前,播放器須要把必需的頭文件所有解析並存入內存,好比以前說的採樣索引表。通常播放器在解析完畢後,會構建三個個表,一個存放時間對應採樣索引,一個存放採樣索引對應在mp4文件中的起始位置(以字節爲單位),一個存放採樣索引對應大小(以字節爲單位)。如下圖爲例

假設播放器須要從第1微秒開始播放,那麼須要把第1微秒的數據放入渲染器。因此會查找下面這三個表。

經過表1,咱們知道該微秒對應第1個採樣(sample),從第一個和第二個表咱們知道,第1個採樣的數據範圍(在mp4文件內)是從第0字節到300(0+300)字節,那麼播放器就會去讀取這個範圍的數據而且放入渲染器中進行渲染。

同時,加載器會基於當前已經緩存的數據,決定是否還須要不停的讀取數據進入內存。通常來講每一個播放器都有默認的緩存值,也會有一個基準線,只有當緩存足夠數據才能放進渲染器進行渲染。

最後同理,當咱們拖動滑動控制器(SeekBar)想快進的時候,咱們和第一步同樣,經過咱們想滑動的時間獲取採樣的索引,再從新開始讀取數據。

綜上所述,播放器在正式播放視頻文件以前,必需要把頭文件所有讀取並解析(這會是一段很是耗時的程序),這也是在線視頻播放的等待時間的瓶頸。在接下來的章節我會介紹自適應視頻播放(Adaptive Streaming),這個技術的發明使得了分段式mp4文件(Fragmented Mp4)技術得以誕生,大大的減小了在線視頻播放的等待時間。

4.在線視頻播放的技術基礎(online video streaming)

在線視頻的播放其實和播放本地視頻的局別就是Extractor讀取的Source,數據源不同,在線播放須要下載數據到內存,再交由Extractor讀取分析。可是既然是在線視頻播放,咱們確定不能把整個容器文件下載到內存或者硬盤再開始解析播放。咱們但願能控制下載的進度,好比我當前在看第10s的視頻內容,因此我只想緩存/下載視頻內容到第20s的位置。

咱們俗稱的漸進式下載(Progressive Downloading)就解決了這一難題。

說的好像是很嚇人的黑科技啊!!!!

其實就是HTTP1.1協議支持的分段式下載而已。。。。。

在HTTP請求裏面假如一個叫RANGE的header,放入起始字節和結束字節,就能夠只下載對應部分的數據,這一header的支持也是各類下載軟件實現斷點下載的基礎。每次斷網的時候記錄下來已經下載的數據的字節數,下次再下載的時候從字節數+1處從新下載而且寫入原有文件就能夠了。


分割線

因此此次分享就結束啦,下一期分享我會開始進入正題,在安卓平臺裏面,對視頻播放的支持,像api啊等等,以及其變遷歷史。

週末愉快!

part3-安卓的MediaCodec API

相關文章
相關標籤/搜索