avformat_seek_file函数介绍
在做音視頻數(shù)據(jù)分析的時候,經(jīng)常會遇到這樣的需求,每隔5分鐘抽取一幀數(shù)據(jù)進行分析。
在做播放器開發(fā)的時候,也會遇到這種情況,就是拖動進度條跳轉(zhuǎn)到某個位置進行播放。
如果直接用?av_read_frame()?不斷讀數(shù)據(jù),讀到第 5 分鐘的?AVPacket?才開始處理,其他讀出來的?AVPacket?丟棄,這樣做會帶來非常大的磁盤IO。
其實上面兩種場景,都可以用同一個函數(shù)解決,那就是?avformat_seek_file(),這個函數(shù)類似于?Linux?的?lseek()?,設(shè)置文件的讀取位置。
只不過?avformat_seek_file()?是用于音視頻文件的。
avformat_seek_file()?函數(shù)的定義如下:
int avformat_seek_file(AVFormatContext *s, int stream_index, int64_t min_ts, int64_t ts, int64_t max_ts, int flags);參數(shù)解釋如下:
1,AVFormatContext *s,已經(jīng)打開的容器示例。
2,int stream_index,流索引,但是只有在?flags?包含?AVSEEK_FLAG_FRAME?的時候才是?設(shè)置某個流的讀取位置。其他情況都只是把這個流的 time_base (時間基)作為參考。
3,int64_t min_ts,跳轉(zhuǎn)到的最小的時間,但是這個變量不一定是時間單位,也有可能是字節(jié)單位,也可能是幀數(shù)單位(第幾幀)。
4,int64_t ts,要跳轉(zhuǎn)到的讀取位置,單位同上。
5,int64_t max_ts,跳轉(zhuǎn)到的最大的時間,單位同上,通常填?INT64_MAX?即可。
6,int flags,跳轉(zhuǎn)的方式,有 4 個?flags,如下:
- AVSEEK_FLAG_BYTE,按字節(jié)大小進行跳轉(zhuǎn)。
- AVSEEK_FLAG_FRAME,按幀數(shù)大小進行跳轉(zhuǎn)。
- AVSEEK_FLAG_ANY,可以跳轉(zhuǎn)到非關(guān)鍵幀的讀取位置,但是解碼會出現(xiàn)馬賽克。
- AVSEEK_FLAG_BACKWARD,往?ts?的后面找關(guān)鍵幀,默認是往?ts?的前面找關(guān)鍵幀。
avformat_seek_file()?函數(shù)默認是把文件的讀取位置,設(shè)置到離?ts?參數(shù)最近的關(guān)鍵幀的地方。
而且默認情況,是容器里面所有流的讀取位置都會被設(shè)置,包括 音頻流,視頻流,字幕流。
只要流的?discard?屬性小于?AVDISCARD_ALL?就會被設(shè)置。
AVStream.discard < AVDISCARD_ALLmin_ts?跟?max_ts?變量有一些設(shè)置的技巧。
如果是快進的時候,min_ts?可以設(shè)置得比 當前位置 大一點,例如加 2。 而?max_ts?可以填?INT64_MAX
min_ts = 當前位置 + 2 max_ts = INT64_MAX+2?是為了防止某些情況,avformat_seek_file()?會把讀取位置往后挪一點。
如果是后退的時候,min_ts?可以填 INT64_MIN,max_ts?可以設(shè)置得比 當前位置 小一點,例如減 2。
min_ts = INT64_MIN max_ts = 當前位置 - 2-2?是為了防止某些情況,avformat_seek_file()?會把讀取位置往前挪一點。
當?flags?為 0 的時候,默認情況,是按時間來?seek?的,而時間基是根據(jù)?stream_index?來確定的。
如果?stream_index?為?-1?,那?ts?的時間基就是?AV_TIME_BASE,
如果stream_index?不等于?-1?,那?ts?的時間基就是?stream_index?對應(yīng)的流的時間基。
這種情況,avformat_seek_file()?會導(dǎo)致容器里面所有流的讀取位置都發(fā)生跳轉(zhuǎn),包括音頻流,視頻流,字幕流。
當?flags?包含?AVSEEK_FLAG_BYTE,ts?參數(shù)就是字節(jié)大小,代表?avformat_seek_file()?會把讀取位置設(shè)置到第幾個字節(jié)。用?av_read_frame()?讀出來的?pkt?里面有一個字段?pos,代表當前讀取的字節(jié)位置。可以用pkt->pos?輔助設(shè)置?ts?參數(shù),
AVSEEK_FLAG_BYTE?是否是對所有流都生效,我后面測試一下再補充。
當?flags?包含?AVSEEK_FLAG_FRAME,ts?參數(shù)就是幀數(shù)大小,代表?avformat_seek_file()?會把讀取位置設(shè)置到第幾幀。這時候?stream_index?可以指定只設(shè)置某個流的讀取位置,如果?stream_index?為?-1?,代表設(shè)置所有的流。
當?flags?包含?AVSEEK_FLAG_ANY,那就代表?seek?可以跳轉(zhuǎn)到非關(guān)鍵幀的位置,但是非關(guān)鍵幀解碼會出現(xiàn)馬賽克。如果不設(shè)置?AVSEEK_FLAG_ANY, 默認是跳轉(zhuǎn)到離?ts?最近的關(guān)鍵幀的位置的。
當?flags?包含?AVSEEK_FLAG_BACKWARD,代表?avformat_seek_file()?在查找里?ts?最近的關(guān)鍵幀的時候,會往?ts?的后面找,默認是往?ts?的前面找關(guān)鍵幀。
提醒:AVSEEK_FLAG_BYTE?,AVSEEK_FLAG_FRAME,AVSEEK_FLAG_ANY?這 3 種方式,有些封裝格式是不支持的。
下面通過一個例子來演示?avformat_seek_file()?函數(shù)的用法。代碼下載地址:GitHb,編譯環(huán)境是 Qt 5.15.2 跟 MSVC2019_64bit 。
運行結(jié)果如下:
可以看到,跳轉(zhuǎn)之后,后面?av_read_frame()?讀取到的?AVPacket?的?pts?跟?pos?都有很大的偏移了。
avformat_seek_file()?函數(shù)介紹完畢。
擴展知識:avformat_seek_file()?對應(yīng)的舊版函數(shù)是?av_seek_frame()
推薦一個零聲學(xué)院免費公開課程,個人覺得老師講得不錯,分享給大家:
Linux,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒體,CDN,P2P,K8S,Docker,TCP/IP,協(xié)程,DPDK等技術(shù)內(nèi)容,立即學(xué)習(xí)
總結(jié)
以上是生活随笔為你收集整理的avformat_seek_file函数介绍的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 怎么在抖音中一键复制微信号打开微信引流
- 下一篇: AWS上配置Cisco ASAv Any