1

私はffmpegで再び問題を抱えています、私はffmpegの初心者です、そして私は最新の良いチュートリアルを見つけることができません...

今回、ffmpegで動画を再生すると、再生速度が速すぎます。ffmpegはFPSを無視しています。動画のFPSが異なるため、スレッドスリープで処理したくありません。

スレッドを作成しました。そこにループがあります。

AVPacket framepacket;

while(av_read_frame(formatContext,&framepacket)>= 0){
    pausecontrol.lock();

    // Is it a video or audio frame¿?
    if(framepacket.stream_index==gotVideoCodec){
        int framereaded;
        // Video? Ok
        avcodec_decode_video2(videoCodecContext,videoFrame,&framereaded,&framepacket);
        // Yeah, did we get it?
        if(framereaded && doit){
            AVRational millisecondbase = {1,1000};
            int f_number = framepacket.dts;
            int f_time = av_rescale_q(framepacket.dts,formatContext->streams[gotVideoCodec]->time_base,millisecondbase);
            currentTime=f_time;
            currentFrameNumber=f_number;

            int stWidth = videoCodecContext->width;
            int stHeight = videoCodecContext->height;
            SwsContext *ctx = sws_getContext(stWidth, stHeight, videoCodecContext->pix_fmt, stWidth,
            stHeight, PIX_FMT_RGB24, SWS_BICUBIC, NULL, NULL, NULL);
            if(ctx!=0){
            sws_scale(ctx,videoFrame->data,videoFrame->linesize,0,videoCodecContext->height,videoFrameRGB->data,videoFrameRGB->linesize);
            QImage framecapsule=QImage(stWidth,stHeight,QImage::Format_RGB888);

            for(int y=0;y<stHeight;y++){
                memcpy(framecapsule.scanLine(y),videoFrameRGB->data[0]+y*videoFrameRGB->linesize[0],stWidth*3);
            }
            emit newFrameReady(framecapsule);
            sws_freeContext(ctx);
            }

        }
    }
    if(framepacket.stream_index==gotAudioCodec){
        // Audio? Ok
    }
    pausecontrol.unlock();
    av_free_packet(&framepacket);
}

何か案が?

4

1 に答える 1

3

最も簡単な解決策は、FPS 値に基づく遅延を使用することです。

firstFrame = true;
for(;;)
{
  //  decoding, color conversion, etc.

  if (!firstFrame)
  {
    const double frameDuration = 1000.0 / frameRate;
    duration_t actualDelay = get_local_time() - lastTime;
    if (frameDuration > actualDelay)
      sleep(frameDuration - actualDelay); 
  }
  else
    firstFrame = false;

  emit newFrameReady(framecapsule);

  lastTime = get_local_time();
}

get_local_time()そしてduration_t抽象的です。

より正確な方法は、各フレームのタイムスタンプを使用することですが、考え方は同じです

于 2012-11-22T10:27:47.810 に答える