基于FFmpeg的视频软解
Posted huluwa508
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了基于FFmpeg的视频软解相关的知识,希望对你有一定的参考价值。
首先需要创建一个AVFormatContext对象,其包含了很多视频的基本信息;
AVFormatContext* m_pFmtCtx; m_pFmtCtx = avformat_alloc_context();
打开视频源,可以通过rtsp协议,也可以直接打开本地视频文件,或者读取内存中的数据。
通过rtsp协议:
// 主码流 const char * rtsp = "rtsp://admin:密码@视频源IP地址:554/h264/ch1/main/av_stream"; // 子码流 //const char * rtsp = "rtsp://admin:密码@视频源IP地址:554/h264/ch1/sub/av_stream"; AVDictionary* options = NULL; av_dict_set(&options, "stimeout", "20000", 0); // 连接超时 av_dict_set(&options, "rtsp_transport", "tcp", 0); // 设置tcp连接,默认为udp,在网络环境不好的情况下可能会丢包 // 打开视频源 avformat_open_input(&m_pFmtCtx, rtsp, NULL, &options);
打开本地文件:
const char * fileName = "C://localfile.mp4"; // 打开视频源 avformat_open_input(&m_pFmtCtx, fileName, NULL, NULL);
读取内存中的数据:
unsigned char * pIOBuffer = (unsigned char *)av_malloc(32768); // 其它大小也是可行的 // 第一个参数是为AVIOContext申请的内存地址 // 第二个参数是每次读取数据的大小,如无要求一般设为4kb // 第三个参数是buffer是否可写 // 第四个参数是refilling(填写)buffer数据回调函数 // 第五个参数是将buffer写入磁盘的回调函数 // 第六个参数是移动读写指针的位置回调函数 AVIOContext * pAVIO = avio_alloc_context(pIOBuffer, 32768, 0, NULL, ReadData, NULL, NULL); m_pFmtCtx->pb = pAVIO; // 打开视频源 avformat_open_input(&m_pFmtCtx, "", NULL, NULL);
查找视频流
// 解析数据流信息 avformat_find_stream_info(m_pFmtCtx, NULL); // 查找视频流 // AVCodec * m_pDecoder; // 解码器 int m_videoIndex = av_find_best_stream(m_pFmtCtx, AVMEDIA_TYPE_VIDEO, -1, -1, &m_pDecoder, 0); // AVStream * m_pVideoStream; // 视频流 m_pVideoStream = m_pFmtCtx->streams[m_videoIndex];
初始化解码器上下文
// AVCodecContext * m_pDecoderCtx; m_pDecoderCtx = avcodec_alloc_context3(m_pDecoder); // 初始化 avcodec_parameters_to_context(m_pDecoderCtx, m_pVideoStream->codecpar); // 打开解码器 avcodec_open2(m_pDecoderCtx, m_pDecoder, NULL);
获取视频帧率(可选,rtsp不需要)
m_pDecoderCtx->framerate = av_guess_frame_rate(m_pFmtCtx, m_pVideoStream, NULL);
循环解码
AVPacket packet = { 0 }; while (av_read_frame(m_pFmtCtx, &packet) >= 0) { if (m_videoIndex == packet.stream_index) { Decode(m_pDecoderCtx, &packet); av_packet_unref(&packet); } } packet.data = NULL; packet.size = 0; // 清空buffer里的残留 Decode(m_pDecoderCtx, &packet); av_packet_unref(&packet); void Decode(AVCodecContext * pDecodeCtx, AVPacket * pPacket) { if (avcodec_send_packet(pDecodeCtx, pPacket) < 0) { return; } while (TRUE) { if (avcodec_receive_frame(pDecodeCtx, m_pFrame) != 0) { break; } // 解码完成 } }
以上是关于基于FFmpeg的视频软解的主要内容,如果未能解决你的问题,请参考以下文章
QT软件开发-基于FFMPEG设计视频播放器-支持软解与硬解-完整例子
QT软件开发-基于FFMPEG设计视频播放器-支持软解与硬解
QT软件开发-基于FFMPEG设计视频播放器-支持软解与硬解
QT软件开发-基于FFMPEG设计视频播放器-支持软解与硬解