解码(四):avcodec_send_packet和avcodec_receive_frame函数讲解

本文详细介绍了FFmpeg中avcodec_send_packet与avcodec_receive_frame两个核心API的使用方法及工作原理。通过示例代码展示了如何发送AVPacket到解码器,并从解码器接收已解码的AVFrame。
摘要由CSDN通过智能技术生成

1.avcodec_send_packet
int avcodec_send_packet(AVCodecContext *avctx,const AVPacket *avpkt);
第一个参数:原来解封装的时候有一个上下文,现在做解码有另外一个上下文,这个上下文贯穿于整个解码逻辑。
第二个参数:会把这个AVPacket放到我们的缓存中。

2.avcodec_receive_frame
int avcodec_receive_frame(AVCodecContext *avctx,AVFrame *frame);
第一个参数:同上。第二个参数:我们在做av_read_packet的时候,讲了AVPacket是由调用者分配的。AVFrame会在每次调用的时候把上次调用的空间清掉,然后重新分配。所以这块可以一直穿同一个对象。

总结:第一个接口只做一件事情,就是把packet写到一个解码队列中。
第二个接口做的事是从已经解码成功的数据中取出一个frame。
注意一点:我们第二个接口时立即获取已经解码的数据,而第一个接口取解码数据的时候因为在队列中有可能无法立即获取。所以前几帧可能会失败。send一次只能发一个包,不代表receive只能收到一个包。第二个接口一次可以receive多个包
看如下代码:

///解码视频
		//发送packet到解码线程  send传NULL后调用多次receive取出所有缓冲帧
		re = avcodec_send_packet(cc, pkt);
		//释放,引用计数-1 为0释放空间
		av_packet_unref(pkt);

		if (re != 0)
		{
			char buf[1024] = { 0 };
			av_strerror(re, buf, sizeof(buf) - 1);
			cout << "avcodec_send_packet  failed! :" << buf << endl;
			continue;
		}

		for(;;)
		{
			//从线程中获取解码接口,一次send可能对应多次receive
			re = avcodec_receive_frame(cc,frame);
			if (re != 0) break;
			cout << "recv frame " << frame->format << " " << frame->linesize[0] << endl;
		}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值