解决ffmpeg中的时间戳同步问题

PTS/DTS(时间戳)

要想解决时间同步问题就必须要了解ffmpeg中的PTS和DTS到底是什么

  • PTS:

    PTS(Presentation TimeStamp)是渲染用的时间戳,播放器会根据这个时间戳进行渲染播放

  • DTS:

    DTS(Decoding TimeStamp)解码时间戳,在视频packet进行解码成frame的时候会使用到

  • 有了PTS为什么还需要DTS

    就拿编码H264来说,H264编码分为I帧,B帧,P帧,I帧是关键帧,也就是一个GOP的最开始帧,B帧是前后参考帧,它属于帧间压缩技术,B帧会通过记录前后两帧进行压缩,P帧是向前参考帧,比B帧的压缩率要低

    因为B帧要参考前后帧,那么在编码之后的帧的时间顺序就会发生变化,所以在没有B帧的时候PTS和DTS应该是一样的,有了B帧之后PTS和DTS也就会发生变化

时间基

有了时间戳之后,还需要将PTS的时间戳转成以秒为单位的时间。这里就需要用到ffmpeg的时间基来进行计算了

先了解一下tbr,tbn,tbc

  • tbr: 是通常说的帧率
  • tbn: 视频流的时间基
  • tbc: 时间解码的时间基

在ffmpeg中,不同的时间戳对应不同的时间基。一般在视频进行渲染的时候渲染的时候用到的就是视频流的时间基tbn,视频流的时间基和帧率有关,每秒30帧,代表一帧需要1/30秒

  • ffmpeg内部有自己的时间基,分别定义了

      // ffmpeg中内部的时间基
      #define		AV_TIME_BASE   1000000
    
      // ffmpeg中分数所表示法
      #define 	AV_TIME_BASE_Q (AVRatonal){1, AV_TIME_BASE}
    
  • av_q2d,计算真实秒数

    // ffmpeg 中进行转换,将时间转成秒,av_q2d可以将时间基转换成double类型小数
    typedef struct AVRational{
    int num; //numerator
    int den; //denominator
    } AVRational;
    static inline double av_q2d(AVRational a){
    /**
    * Convert rational to double.
    * @param a rational to convert
    **/
        return a.num / (double) a.den;
    }
    // 通过pts*时间基的小数可以算出当前的流的秒数
    AVRational time_base = {1, 30};
    double time_sec = pts * av_q2d(time_base)
    
    
  • av_rescale_q(),不同的时间基直接的转换

    /**
     * Rescale a 64-bit integer by 2 rational numbers.
     *
     * The operation is mathematically equivalent to `a * bq / cq`.
     *
     * This function is equivalent to av_rescale_q_rnd() with #AV_ROUND_NEAR_INF.
     *
     * @see av_rescale(), av_rescale_rnd(), av_rescale_q_rnd()
     */
    int64_t av_rescale_q(int64_t a, AVRational bq, AVRational cq) av_const;
    
    /*
    @param a : 需要进行转换的时间戳
    @bq	: 对应转换的时间戳的原来的时间基
    @cq	: 转换之后的时间基
    
    */
    

总结

通过ffmpeg的时间戳和时间基的使用,可以有效的解决音视频的同步的问题

posted @ 2020-01-16 06:46  FANDX  阅读(2551)  评论(0编辑  收藏  举报