Qt+FFmpeg+opengl从零制作视频播放器-3.解封装

时间:2024-03-14 16:24:47

解封装:如下图所示,就是将FLV、MKV、MP4等文件解封装为视频H.264或H.265压缩数据,音频MP3或AAC的压缩数据,下图为常用的基本操作。

 ffmpeg使用解封装的基本流程如下:

在使用FFmpeg API之前,需要先注册API,然后才能使用API。当然,新版本ffmpeg库不需要再调用下面的方法。

av_register_all()

初始化网络配置

avformat_network_init

设置一些参数,如下图所示,设置了最大延迟、传输协议等参数。

	//参数设置
	AVDictionary *opts = NULL;

	//设置rtsp流已tcp协议打开
	av_dict_set(&opts, "rtsp_transport", "tcp", 0);
	av_dict_set(&opts, "max_delay", "500", 0);
	av_dict_set(&opts, "buffer_size", "1024000", 0);
	av_dict_set(&opts, "probsize", "4096", 0);
	av_dict_set(&opts, "fps", "25", 0);

构建AVFormatContext,声明输入的封装结构体,通过输入文件或者流地址作为封装结构的句柄。

    AVFormatContext* inputFmtCtx = nullptr;
    const char* inputUrl = "test.mp4";
 
    ///打开输入的流,获取数据 begin
    int ret = avformat_open_input(&inputFmtCtx, inputUrl, NULL, NULL);

查找音视频流信息,通过下面的接口与AVFormatContext中建立输入文件对应的流信息。

    //查找;
    if (avformat_find_stream_info(inputFmtCtx, NULL) < 0)
    {
        printf("Couldn't find stream information.\n");
        return false;
    }


读取音视频流,采用av_read_frame来读取数据包,读出来的数据存储在AVPacket中,确定其为音频、视频、字幕数据,最后解码,或者存储。

    AVPacket* pkt = NULL;
    pkt = av_packet_alloc();
 
    while (av_read_frame(inputFmtCtx, pkt) >= 0)
    {
        //获取数据包
 
        //.....解码或者存储
 
        //重置
        av_packet_unref(pkt);
    }

上述代码所示,通过循环调用av_read_frame()读取到pkt包,然后可以进行解码或者存储数据,如果读取的数据结束,则退出循环,开始指向结束操作。

每次使用完AVPacket包后,需要重置,否则会内存泄漏。

    //重置
    av_packet_unref(pkt);

执行结束后关闭输入文件,释放资源。

    //关闭
    avformat_close_input(&inputFmtCtx);

    //释放
    avformat_free_context(inputFmtCtx);
 
    //释放资源
    av_packet_free(&pkt);

 新建空白解决方案,如下图所示。

解决方案,右键-添加-新建项目。 

新建Qt控制台程序,3_demuxDemo 

然后,配置ffmpeg的编译环境:看目录4。

ffmpeg环境配置 

解封装源码示例:

#include <QtCore/QCoreApplication>

extern "C" {
#include "libavutil/avstring.h"
#include "libavutil/channel_layout.h"
#include "libavutil/eval.h"
#include "libavutil/mathematics.h"
#include "libavutil/pixdesc.h"
#include "libavutil/imgutils.h"
#include "libavutil/dict.h"
#include "libavutil/fifo.h"
#include "libavutil/parseutils.h"
#include "libavutil/samplefmt.h"
#include "libavutil/time.h"
#include "libavutil/bprint.h"
#include "libavutil/opt.h"
#include "libavformat/avformat.h"
#include "libavcodec/avcodec.h"
#include "libavfilter/avfilter.h"
#include "libavdevice/avdevice.h"
#include "libswscale/swscale.h"
#include "libavutil/opt.h"
#include "libavutil/imgutils.h"  
#include "libavcodec/avfft.h"
#include "libswresample/swresample.h"
}

#include <iostream>
using namespace std;

static double r2d(AVRational r)
{
	return r.den == 0 ? 0 : (double)r.num / (double)r.den;
}

int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);

	//av_register_all();
	avformat_network_init();

	AVDictionary* options = NULL;

	av_dict_set(&options, "buffer_size", "1024000", 0);
	av_dict_set(&options, "max_delay", "500000", 0);
	av_dict_set(&options, "stimeout", "2000000", 0);
	av_dict_set(&options, "rtsp_transport", "tcp", 0);

	AVFormatContext* inputFmtCtx = nullptr;
	const char* inputUrl = "F:/1920x1080.mp4";

	///打开输入的流,获取数据 begin
	int ret = avformat_open_input(&inputFmtCtx, inputUrl, NULL, NULL);
	if (ret != 0)
	{
		printf("Couldn't open input stream.\n");
		return -1;
	}

	int vIndex = -1;
	int aIndex = -1;

	//查找;
	if (avformat_find_stream_info(inputFmtCtx, NULL) < 0)
	{
		printf("Couldn't find stream information.\n");
		return false;
	}


	for (int i = 0; i < inputFmtCtx->nb_streams/*视音频流的个数*/; i++)
	{
		//查找视频
		if (inputFmtCtx->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_VIDEO)
		{
			vIndex = i;
		}
		else if (inputFmtCtx->streams[i]->codecpar->codec_type == AVMEDIA_TYPE_AUDIO)
		{
			aIndex = i;
		}
	}

	//===================video=================
	//视频宽
	int width = inputFmtCtx->streams[vIndex]->codecpar->width;

	//视频高
	int height = inputFmtCtx->streams[vIndex]->codecpar->height;

	//视频总时长s
	int64_t m_totalTime = static_cast<double>(inputFmtCtx->duration) / AV_TIME_BASE;

	//获取帧率;
	int fps = r2d(inputFmtCtx->streams[vIndex]->avg_frame_rate);
	if (fps == 0)
	{
		fps = 25;
	}

	int iHour, iMinute, iSecond, iTotalSeconds;//HH:MM:SS
	//打印结构体信息
	puts("AVFormatContext信息:");
	puts("---------------------------------------------");
	iTotalSeconds = (int)inputFmtCtx->duration/*微秒*/ / 1000000;
	iHour = iTotalSeconds / 3600;//小时
	iMinute = iTotalSeconds % 3600 / 60;//分钟
	iSecond = iTotalSeconds % 60;//秒
	printf("持续时间:%02d:%02d:%02d\n", iHour, iMinute, iSecond);
	printf("平均混合码率:%d kb/s\n", inputFmtCtx->bit_rate / 1000);
	printf("视音频个数:%d\n", inputFmtCtx->nb_streams);
	puts("---------------------------------------------");

	puts("AVInputFormat信息:");
	puts("---------------------------------------------");
	printf("封装格式名称:%s\n", inputFmtCtx->iformat->name);
	printf("封装格式长名称:%s\n", inputFmtCtx->iformat->long_name);
	printf("封装格式扩展名:%s\n", inputFmtCtx->iformat->extensions);
	printf("封装格式ID:%d\n", inputFmtCtx->iformat->raw_codec_id);
	puts("---------------------------------------------");

	puts("AVStream信息:");
	puts("---------------------------------------------");
	printf("视频流标识符:%d\n", inputFmtCtx->streams[vIndex]->index);
	printf("音频流标识符:%d\n", inputFmtCtx->streams[aIndex]->index);
	printf("视频流长度:%d微秒\n", inputFmtCtx->streams[vIndex]->duration);
	printf("音频流长度:%d微秒\n", inputFmtCtx->streams[aIndex]->duration);
	puts("---------------------------------------------");

	printf("视频时长:%d微秒\n", inputFmtCtx->streams[vIndex]->duration);
	printf("音频时长:%d微秒\n", inputFmtCtx->streams[aIndex]->duration);
	printf("视频宽:%d\n", inputFmtCtx->streams[vIndex]->codecpar->width);
	printf("视频高:%d\n", inputFmtCtx->streams[vIndex]->codecpar->height);
	printf("音频采样率:%d\n", inputFmtCtx->streams[aIndex]->codecpar->sample_rate);
	printf("音频信道数目:%d\n", inputFmtCtx->streams[aIndex]->codecpar->channels);

	puts("AVFormatContext元数据:");
	puts("---------------------------------------------");
	AVDictionaryEntry *dict = NULL;
	while (dict = av_dict_get(inputFmtCtx->metadata, "", dict, AV_DICT_IGNORE_SUFFIX))
	{
		printf("[%s] = %s\n", dict->key, dict->value);
	}
	puts("---------------------------------------------");

	puts("AVStream视频元数据:");
	puts("---------------------------------------------");
	dict = NULL;
	while (dict = av_dict_get(inputFmtCtx->streams[vIndex]->metadata, "", dict, AV_DICT_IGNORE_SUFFIX))
	{
		printf("[%s] = %s\n", dict->key, dict->value);
	}
	puts("---------------------------------------------");

	puts("AVStream音频元数据:");
	puts("---------------------------------------------");
	dict = NULL;
	while (dict = av_dict_get(inputFmtCtx->streams[aIndex]->metadata, "", dict, AV_DICT_IGNORE_SUFFIX))
	{
		printf("[%s] = %s\n", dict->key, dict->value);
	}
	puts("---------------------------------------------");


	av_dump_format(inputFmtCtx, -1, inputUrl, 0);
	printf("\n\n编译信息:\n%s\n\n", avcodec_configuration());

	AVPacket* pkt = NULL;
	pkt = av_packet_alloc();

	while (av_read_frame(inputFmtCtx, pkt) >= 0)
	{
		//获取数据包

		//.....解码或者存储

		//重置
		av_packet_unref(pkt);
	}


	//关闭
	avformat_close_input(&inputFmtCtx);

	//释放
	avformat_free_context(inputFmtCtx);

	//释放资源
	av_packet_free(&pkt);


    return a.exec();
}

运行截图:

完整工程:

https://download.csdn.net/download/wzz953200463/88959152https://download.csdn.net/download/wzz953200463/88959152