使用 ffmpeg API 进行 MPEG-2 视频编码:文件大小比 ffmpeg.exe 更大
我正在尝试使用 ffmpeg API 将视频从原始 YUV 编码为 MPEG-2。
我的问题是 API 生成的文件大约是。比 ffmpeg 本身生成的等效文件大 1.7 倍。
我使用量化参数(通过 qmin
和 qmax
)而不是比特率。
API-版本基本上是:
//...
pCodecCtx->pix_fmt = PIX_FMT_YUV420P;
pCodecCtx->qmin = 3;
pCodecCtx->qmax = 3;
pCodecCtx->time_base.num = 1;
pCodecCtx->time_base.den = 30;
avcodec_open(pCodecCtx, avcodec_find_encoder(CODEC_ID_MPEG2VIDEO));
//...
while(/*...*/) {
avcodec_encode_video(pCodecCtx, pOutbuf, outbufSize, pPicture);
//..
}
//...
对于ffmpeg本身,我使用命令:
ffmpeg -s 352x288 -r 30 -i foreman_352x288.yuv -f mpeg2video -vcodec mpeg2video -r 30 -pix_fmt yuv420p -qmin 3 -qmax 3 foreman.m2v
为什么API-生成文件达到5212 kb的比特率/s 和 ffmpeg 为相同的 qp
生成的文件,比特率为 3047 kb/s?
(更令人费解的是,较小的 ffmpeg 版本的 PSNR 略高,分别为 40.49 dB 和 40.02 dB)。
还有其他我遗漏的相关参数吗? ffmpeg真的尊重量化参数吗?
I am trying to encode a video from raw YUV to MPEG-2 using the ffmpeg API.
My problem is that the API-generated file is approx. 1.7 times bigger than the equivalent files generated by ffmpeg itself.
I use the quantization parameter (via qmin
and qmax
) instead of the bitrate.
The API-version is basically:
//...
pCodecCtx->pix_fmt = PIX_FMT_YUV420P;
pCodecCtx->qmin = 3;
pCodecCtx->qmax = 3;
pCodecCtx->time_base.num = 1;
pCodecCtx->time_base.den = 30;
avcodec_open(pCodecCtx, avcodec_find_encoder(CODEC_ID_MPEG2VIDEO));
//...
while(/*...*/) {
avcodec_encode_video(pCodecCtx, pOutbuf, outbufSize, pPicture);
//..
}
//...
For ffmpeg itself, I use the command:
ffmpeg -s 352x288 -r 30 -i foreman_352x288.yuv -f mpeg2video -vcodec mpeg2video -r 30 -pix_fmt yuv420p -qmin 3 -qmax 3 foreman.m2v
Why does the API-generate file achieve a bitrate of 5212 kb/s and the file generated by ffmpeg for the same qp
a bitrate of 3047 kb/s??
(Even more puzzling is that the smaller ffmpeg version has a slightly higher PSNR, 40.49 dB vs. 40.02 dB).
Are there any other relevant parameters that I missed? Does the ffmpeg actually respect the quantization parameter?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(1)
使用ffmpeg API时,需要为每一帧手动设置图片类型(I帧、P帧等)。默认情况下,ffmpeg 将使每一帧成为 I 帧。
解决方案是在编码帧之前设置图片类型(此处 GOP 大小为 12):
请注意,在编码之前设置
pCodecCtx->gop_size
没有帮助。When using the ffmpeg API, the picture type (I-frame, P-frame, etc.) needs to be set manually for each frame. By default, ffmpeg will make every frame an I-frame.
The solution is to set the picture type before encoding a frame (here for a GOP size of 12):
Note that setting
pCodecCtx->gop_size
before the encoding does not help.