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

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;
		}

你可能感兴趣的:(音视频与流媒体,音视频)