ffmpeg封装和解封装介绍-(10)综合完成视频重编码为h265,解封装解码编码再封装

主函数逐句解析:

由于代码太多我们只解析主函数,(其他封装函数见前面文章,同时用到了解码编码封装代码)。

初始化和参数处理 

int main(int argc, char* argv[])
{/// 输入参数处理string useage = "124_test_xformat 输入文件 输出文件 开始时间(秒) 结束时间(秒) 视频宽 视频高\n";useage += "124_test_xformat v1080.mp4 test_out.mp4 10 20 400 300";cout << useage << endl;if (argc < 3){return -1;}string in_file = argv[1];//输入文件参数string out_file = argv[2];//输出文件参数

 这段代码是程序的入口。它首先定义了程序的用法提示,并将其打印出来。然后,它检查命令行参数的数量是否正确,若不足三个参数,则程序退出。接着,它从命令行参数中获取输入文件和输出文件的路径。

截取时间参数和视频尺寸参数 

    /// 截取10 ~ 20 秒之间的音频视频 取多不取少// 假定 9 11秒有关键帧 我们取第9秒int begin_sec = 0;    //截取开始时间int end_sec = 0;      //截取结束时间if (argc > 3)begin_sec = atoi(argv[3]);if (argc > 4)end_sec = atoi(argv[4]);int video_width = 0;int video_height = 0;if (argc > 6)video_width = atoi(argv[5]);video_height = atoi(argv[6]);

 这段代码获取截取的开始和结束时间(以秒为单位),以及视频的宽度和高度。如果命令行参数中没有提供这些参数,则使用默认值。

解封装输入文件 

    /// 解封装//解封装输入上下文XDemux demux;AVFormatContext* demux_c = demux.Open(in_file.c_str());demux.set_c(demux_c);long long video_begin_pts = 0;long long audio_begin_pts = 0;  //音频的开始时间long long video_end_pts = 0;//开始截断秒数 算出输入视频的pts//if (begin_sec > 0){//计算视频的开始和结束播放pts if (demux.video_index() >= 0 && demux.video_time_base().num > 0){double t = (double)demux.video_time_base().den / (double)demux.video_time_base().num;video_begin_pts = t * begin_sec;video_end_pts = t * end_sec;demux.Seek(video_begin_pts, demux.video_index()); //移动到开始帧}//计算音频的开始播放ptsif (demux.audio_index() >= 0 && demux.audio_time_base().num > 0){double t = (double)demux.audio_time_base().den / (double)demux.audio_time_base().num;audio_begin_pts = t * begin_sec;}}

这段代码创建了解封装对象 XDemux 并打开输入文件。它计算开始和结束时间对应的视频和音频PTS(Presentation Timestamps),然后将解封装器定位到视频开始时间的关键帧。 

视频解码器初始化 

    /视频解码器的初始化并打开解码器XDecode decode;AVCodecContext* decode_c = decode.Create(demux.video_codec_id(), false);//设置视频解码器参数demux.CopyPara(demux.video_index(), decode_c);decode.set_c(decode_c);decode.Open();auto frame = decode.CreateFrame(); //解码后存储/

这段代码初始化视频解码器 XDecode 并打开解码器。它从解封装器中复制视频参数,并为解码后的帧数据分配内存。 

视频编码器初始化 

    /视频编码的初始化if (demux.video_index() >= 0){if (video_width <= 0)video_width = demux_c->streams[demux.video_index()]->codecpar->width;if (video_height <= 0)video_height = demux_c->streams[demux.video_index()]->codecpar->height;}XEncode encode;auto encode_c = encode.Create(AV_CODEC_ID_H265, true);encode_c->pix_fmt = AV_PIX_FMT_YUV420P;encode_c->width = video_width;encode_c->height = video_height;encode.set_c(encode_c);encode.Open();/

 这段代码初始化视频编码器 XEncode 并打开编码器。如果没有指定视频宽度和高度,则使用解封装器中的视频参数。编码器设置为 H.265 编码,像素格式为 YUV420P。

封装初始化 

    /// 封装//编码器上下文//const char* out_url = "test_mux.mp4";XMux mux;auto mux_c = mux.Open(out_file.c_str());mux.set_c(mux_c);auto mvs = mux_c->streams[mux.video_index()]; //视频流信息auto mas = mux_c->streams[mux.audio_index()]; //视频流信息//有视频if (demux.video_index() >= 0){mvs->time_base.num = demux.video_time_base().num;mvs->time_base.den = demux.video_time_base().den;//复制视频参数//demux.CopyPara(demux.video_index(), mvs->codecpar);// 复制编码器格式avcodec_parameters_from_context(mvs->codecpar, encode_c);}//有音频if (demux.audio_index() >= 0){mas->time_base.num = demux.audio_time_base().num;mas->time_base.den = demux.audio_time_base().den;//复制音频参数demux.CopyPara(demux.audio_index(), mas->codecpar);}// 写入头部,会改变timebasemux.WriteHead();

这段代码初始化封装器 XMux 并打开输出文件。它设置视频和音频流的时间基,并复制编码器参数到封装器。最后,它写入文件头。

为什么初始化封装器的时候复制编码器参数到封装器而不是解码器参数给封装器

在视频处理的流程中,封装器(muxer)初始化时需要复制编码器的参数,而不是解码器的参数,这是因为封装器的目的是将编码后的数据打包成一个文件格式,而不是处理原始解码后的数据。让我们具体看看原因:

  • 解码器参数:这些参数用于描述如何从压缩格式(如H.264、HEVC等)解码出原始的未压缩数据(如YUV帧)。解码器参数包括压缩格式、比特率、分辨率等,但这些参数主要用于解码过程。

  • 编码器参数:这些参数用于描述如何将原始的未压缩数据(如YUV帧)压缩成目标格式(如H.264、HEVC等)。编码器参数包括目标压缩格式、比特率、分辨率、帧率等。封装器需要这些参数来正确打包和封装编码后的数据流。

  • 封装器的任务是将已经编码的数据按照特定的容器格式(如MP4、MKV等)打包成文件。因此,它需要知道数据的编码方式(即编码器参数)来正确地打包数据流。

读取、解码、编码和写入数据

    int audio_count = 0;int video_count = 0;double total_sec = 0;AVPacket pkt;for (;;){if (!demux.Read(&pkt)){break;}// 视频 时间大于结束时间if (video_end_pts > 0&& pkt.stream_index == demux.video_index()&& pkt.pts > video_end_pts){av_packet_unref(&pkt);break;}if (pkt.stream_index == demux.video_index()) //视频{mux.RescaleTime(&pkt, video_begin_pts, demux.video_time_base());//解码视频if (decode.Send(&pkt)){while (decode.Recv(frame)){// 修改图像尺寸 //视频编码AVPacket* epkt = encode.Encode(frame);if (epkt){epkt->stream_index = mux.video_index();//写入视频帧 会清理pktmux.Write(epkt);//av_packet_free(&epkt);}}}video_count++;if (demux.video_time_base().den > 0)total_sec += pkt.duration * ((double)demux.video_time_base().num / (double)demux.video_time_base().den);av_packet_unref(&pkt);}else if (pkt.stream_index == demux.audio_index()){mux.RescaleTime(&pkt, audio_begin_pts, demux.audio_time_base());audio_count++;//写入音频帧 会清理pktmux.Write(&pkt);}else{av_packet_unref(&pkt);}}

这段代码读取、解码和编码数据。它从输入文件中读取数据包,并根据数据包的类型进行不同的处理。如果是视频数据包,则解码后编码并写入输出文件;如果是音频数据包,则直接写入输出文件。 

写入结尾并释放资源 

    //写入结尾 包含文件偏移索引mux.WriteEnd();demux.set_c(nullptr);mux.set_c(nullptr);encode.set_c(nullptr);cout << "输出文件" << out_file << ":" << endl;cout << "视频帧:" << video_count << endl;cout << "音频帧:" << audio_count << endl;cout << "总时长:" << total_sec << endl;getchar();return 0;
}

运行结果:

以h265重新编码了原视频,生成了一段视频。

 主函数代码总览:


#include <iostream>
#include <thread>
#include "xdemux.h"
#include "xmux.h"
#include "xdecode.h"
#include "xencode.h"
#include "xvideo_view.h"
using namespace std;
extern "C"
{ 
#include <libavformat/avformat.h>
}int main(int argc, char* argv[])
{/// 输入参数处理string useage = "124_test_xformat 输入文件 输出文件 开始时间(秒) 结束时间(秒) 视频宽 视频高\n";useage += "124_test_xformat v1080.mp4 test_out.mp4 10 20 400 300";cout << useage << endl;if (argc < 3){return -1;}string in_file = argv[1];//输入文件参数string out_file = argv[2];//输出文件参数/// 截取10 ~ 20 秒之间的音频视频 取多不取少// 假定 9 11秒有关键帧 我们取第9秒int begin_sec = 0;    //截取开始时间int end_sec = 0;      //截取结束时间if (argc > 3)begin_sec = atoi(argv[3]);if (argc > 4)end_sec = atoi(argv[4]);int video_width = 0;int video_height = 0;if (argc > 6)video_width = atoi(argv[5]);video_height = atoi(argv[6]);/// 解封装//解封装输入上下文XDemux demux;AVFormatContext* demux_c = demux.Open(in_file.c_str());demux.set_c(demux_c);long long video_begin_pts = 0;long long audio_begin_pts = 0;  //音频的开始时间long long video_end_pts = 0;//开始截断秒数 算出输入视频的pts//if (begin_sec > 0){//计算视频的开始和结束播放pts if (demux.video_index() >= 0 && demux.video_time_base().num > 0){double t = (double)demux.video_time_base().den / (double)demux.video_time_base().num;video_begin_pts = t * begin_sec;video_end_pts = t * end_sec;demux.Seek(video_begin_pts, demux.video_index()); //移动到开始帧}//计算音频的开始播放ptsif (demux.audio_index() >= 0 && demux.audio_time_base().num > 0){double t = (double)demux.audio_time_base().den / (double)demux.audio_time_base().num;audio_begin_pts = t * begin_sec;}}/视频解码器的初始化并打开解码器XDecode decode;AVCodecContext* decode_c = decode.Create(demux.video_codec_id(), false);//设置视频解码器参数demux.CopyPara(demux.video_index(), decode_c);decode.set_c(decode_c);decode.Open();auto frame = decode.CreateFrame(); //解码后存储//视频编码的初始化if (demux.video_index() >= 0){if (video_width <= 0)video_width = demux_c->streams[demux.video_index()]->codecpar->width;if (video_height <= 0)video_height = demux_c->streams[demux.video_index()]->codecpar->height;}XEncode encode;auto encode_c = encode.Create(AV_CODEC_ID_H265, true);encode_c->pix_fmt = AV_PIX_FMT_YUV420P;encode_c->width = video_width;encode_c->height = video_height;encode.set_c(encode_c);encode.Open();//// 封装//编码器上下文//const char* out_url = "test_mux.mp4";XMux mux;auto mux_c = mux.Open(out_file.c_str());mux.set_c(mux_c);auto mvs = mux_c->streams[mux.video_index()]; //视频流信息auto mas = mux_c->streams[mux.audio_index()]; //视频流信息//有视频if (demux.video_index() >= 0){mvs->time_base.num = demux.video_time_base().num;mvs->time_base.den = demux.video_time_base().den;//复制视频参数//demux.CopyPara(demux.video_index(), mvs->codecpar);// 复制编码器格式avcodec_parameters_from_context(mvs->codecpar, encode_c);}//有音频if (demux.audio_index() >= 0){mas->time_base.num = demux.audio_time_base().num;mas->time_base.den = demux.audio_time_base().den;//复制音频参数demux.CopyPara(demux.audio_index(), mas->codecpar);}// 写入头部,会改变timebasemux.WriteHead();int audio_count = 0;int video_count = 0;double total_sec = 0;AVPacket pkt;for (;;){if (!demux.Read(&pkt)){break;}// 视频 时间大于结束时间if (video_end_pts > 0&& pkt.stream_index == demux.video_index()&& pkt.pts > video_end_pts){av_packet_unref(&pkt);break;}if (pkt.stream_index == demux.video_index()) //视频{mux.RescaleTime(&pkt, video_begin_pts, demux.video_time_base());//解码视频if (decode.Send(&pkt)){while (decode.Recv(frame)){// 修改图像尺寸 //视频编码AVPacket* epkt = encode.Encode(frame);if (epkt){epkt->stream_index = mux.video_index();//写入视频帧 会清理pktmux.Write(epkt);//av_packet_free(&epkt);}}}video_count++;if (demux.video_time_base().den > 0)total_sec += pkt.duration * ((double)demux.video_time_base().num / (double)demux.video_time_base().den);av_packet_unref(&pkt);}else if (pkt.stream_index == demux.audio_index()){mux.RescaleTime(&pkt, audio_begin_pts, demux.audio_time_base());audio_count++;//写入音频帧 会清理pktmux.Write(&pkt);}else{av_packet_unref(&pkt);}}//写入结尾 包含文件偏移索引mux.WriteEnd();demux.set_c(nullptr);mux.set_c(nullptr);encode.set_c(nullptr);cout << "输出文件" << out_file << ":" << endl;cout << "视频帧:" << video_count << endl;cout << "音频帧:" << audio_count << endl;cout << "总时长:" << total_sec << endl;getchar();return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/351602.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

教育小程序开发:技术实现与实践案例

随着信息技术的不断进步&#xff0c;教育小程序在教育领域的应用越来越广泛。教育小程序开发不仅可以提高教学效率&#xff0c;还能够提供个性化的学习体验。本文将以技术代码为例&#xff0c;详细介绍教育小程序开发的关键技术和实践案例&#xff0c;帮助开发者更好地理解和实…

鸿蒙应用开发

学习视频&#xff1a; 00.课程介绍_哔哩哔哩_bilibili 官网&#xff1a;开发者文档中心 | 华为开发者联盟 (huawei.com) 开发工具 &#xff1a;DevEcoStudio &#xff0c; 类似Jetbrains 全家桶 ArkTS开发语言 &#xff1a;&#xff08;基于TS,集成了前端语言&#xf…

奥特曼谈AI的机遇、挑战与人类自我反思:中国将拥有独特的大语言模型

奥特曼在对话中特别提到&#xff0c;中国将在这个领域扮演重要角色&#xff0c;孕育出具有本土特色的大语言模型。这一预见不仅彰显了中国在全球人工智能领域中日益增长的影响力&#xff0c;也预示着未来技术发展的多元化趋势。 ①奥特曼认为AI在提升生产力方面已显现积极作用&…

【JS重点17】原型链(面试重点)

一&#xff1a;原型链底层原理 以下面一段代码为例&#xff0c;基于原型对象&#xff08;Star构造函数的原型对象&#xff09;的继承使得不同构造函数的原型对象关联在一起&#xff08;此处是最大的构造函数Object原型对象&#xff09;&#xff0c;并且这种关联的关系是一种链…

Project ERROR: Unknown module(s) in QT: xlsx

Qt5下Qxlsx模块安装及使用_qt5xlsx-CSDN博客 主要参考上面这篇文章&#xff01; Perl的安装与配置_perl安装-CSDN博客 1.1 windows环境安装Perl_windows perl-CSDN博客 首先&#xff0c;需要安装Perl,我安装的是Windows版本的。 Download & Install Perl - ActiveStat…

绿色版DirectoryOpus功能强大且高度可定制的Windows文件管理器

Directory Opus&#xff08;通常简称为DOpus&#xff09;是一款功能强大且高度可定制的Windows文件管理器。它提供了许多超越Windows默认文件资源管理器&#xff08;Explorer&#xff09;的功能&#xff0c;使得文件和文件夹的管理变得更加高效和直观。以下是对Directory Opus的…

如何用Java程序实现一个简单的消息队列?

在Java程序中&#xff0c;可以使用内置的java.util.concurrent.BlockingQueue作为消息队列存放的容器&#xff0c;来实现一个简单的消息队列。 具体实现如下&#xff0c;在这个例子中&#xff0c;我们创建了一个生产者线程和一个消费者线程&#xff0c;他们共享同一个阻塞队列…

Nginx配置详细解释:(4)高级配置

目录 1.网页的状态页 2.Nginx第三方模块(echo) 3.变量 4.自定义访问日志 5.Nginx压缩功能 6.https功能 7.自定义图标 Nginx除了一些基本配置外&#xff0c;还有一些高级配置&#xff0c;如网页的状态&#xff0c;第三方模块需要另外安装&#xff0c;支持变量&#xff0c…

使用了代理IP怎么还会被封?代理IP到底有没有效果

代理IP作为一种网络工具&#xff0c;被广泛应用于各种场景&#xff0c;例如网络爬虫、海外购物、规避地区限制等。然而&#xff0c;很多用户在使用代理IP的过程中却发现自己的账号被封禁&#xff0c;这让他们不禁产生疑问&#xff1a;使用了代理IP怎么还会被封&#xff1f;代理…

Unity接入PS5手柄和Xbox手柄以及Android平台的(以及不同平台分析)

Unity接入PS5手柄和Xbox手柄以及Android平台的&#xff08;以及不同平台分析&#xff09; 介绍Unity手柄小知识PC端和编辑器上的摇杆事件和滑动事件PS5手柄Xbox手柄北通手柄 安卓环境下&#xff08;安卓手机或者安卓模拟器&#xff09;PS5手柄Xbox手柄北通手柄 总结 介绍 最近…

Scala网络编程:代理设置与Curl库应用实例

在网络编程的世界里&#xff0c;Scala以其强大的并发模型和函数式编程特性&#xff0c;成为了开发者的得力助手。然而&#xff0c;网络请求往往需要通过代理服务器进行&#xff0c;以满足企业安全策略或访问控制的需求。本文将深入探讨如何在Scala中使用Curl库进行网络编程&…

消息队列-RabbitMQ-延时队列实现

死信队列 DLX,全称为Dead-Letter-Exchange,死信交换机&#xff0c;死信邮箱。当消息在一个队列中变成死信之后&#xff0c;它能重新发送到另外一个交换器中&#xff0c;这个交换器就是DLX&#xff0c;绑定DLX的队列就称为死信队列。 导致死信的几种原因&#xff1a; ● 消息…

数据交换平台_10_activatemq 中间件容错性测试

目录概要 3. 容错测试: - 模拟ActiveMQ在异常情况下的表现,如网络中断、节点故障等。 - 观察ActiveMQ的容错机制是否能够正确处理异常情况,保证消息的可靠传输。 - 根据容错测试结果,优化ActiveMQ的容错机制,确保系统在面对异常情况时能够正确处理并恢复。 设计: 容错测…

ffmpeg解封装rtsp并录制视频-(2)使用VLC模拟一个rtsp服务器并用ffmpeg解封装该rtsp流

VCL模拟服务器并打开播放该视频文件&#xff1a; - 准备好一个mp4文件&#xff0c;打开vlc软件 - 选择“媒体”》“流” - 添加一个mp4文件 - 点击下方按钮选择“串流” - 下一步目标选择rtsp 点击“添加” - 端口默认8554 - 路径设置 /test - 用…

Shell脚本从入门到实战

一、概述 shell 是一个命令行解释器&#xff0c;它接受应用程序、用户命令&#xff0c;然后调用操作系统内核。 shell 还是一个功能强大编程语言&#xff0c;易调试&#xff0c;易编写&#xff0c;灵活性强。 二、mac 怎么重启docker 1.如何重启 Docker on Mac 在 macOS 上…

DockerCompose+Jenkins+Pipeline流水线打包Vue项目(解压安装配置Node)入门

场景 DockerComposeJenkinsPipeline流水线打包SpringBoot项目(解压安装配置JDK、Maven等)入门&#xff1a; DockerComposeJenkinsPipeline流水线打包SpringBoot项目(解压安装配置JDK、Maven等)入门-CSDN博客 以上使用流水线配置和打包springboot后台项目&#xff0c;如果要使…

机器学习(V)--无监督学习(二)主成分分析

当数据的维度很高时&#xff0c;很多机器学习问题变得相当困难&#xff0c;这种现象被称为维度灾难&#xff08;curse of dimensionality&#xff09;。 在很多实际的问题中&#xff0c;虽然训练数据是高维的&#xff0c;但是与学习任务相关也许仅仅是其中的一个低维子空间&am…

springboot优雅shutdown时异步线程安全优化

前面针对graceful shutdown写了两篇文章 第一篇&#xff1a; https://blog.csdn.net/chenshm/article/details/139640775 只考虑了阻塞线程&#xff0c;没有考虑异步线程 第二篇&#xff1a; https://blog.csdn.net/chenshm/article/details/139702105 第二篇考虑了多线程的安全…

智能体(Agent)实战——从gpts到auto gen

一.GPTs 智能体以大模型作为大脑&#xff0c;同时配备技能&#xff0c;使其能够完成具体的任务。同时&#xff0c;为了应用于垂直领域&#xff0c;我们需要为大模型定义一个角色&#xff0c;并构建知识库。最后&#xff0c;定义完整的流程&#xff0c;使其完成整个任务。以组会…

目标检测算法YOLOv10简介

YOLOv10由Ao Wang等人于2024年提出&#xff0c;论文名为&#xff1a;《YOLOv10: Real-Time End-to-End Object Detection》&#xff0c;论文见&#xff1a;https://arxiv.org/pdf/2405.14458 &#xff1b;源码见: https://github.com/THU-MIG/yolov10 以下内容主要来自论文&a…