当前位置: 首页 > news >正文

FFMPEG C++封装(二)

4 详细设计

这章是FFMPEG C++封装库的详细设计。

4.1 Init

该模块初始化FFMPEG库。

4.1.1 Init定义

namespace media
{
namespace sdk
{
void  MEDIASDK_EXPORT Init();
}
}

函数说明:

  • Init 初始化FFMPEG库,该函数可调用多次。

4.1.2 Init实现

namespace media
{
namespace sdk
{
void Init()
{static bool isInit = false;if(isInit)return;isInit = true;avcodec_register_all();avfilter_register_all();av_register_all();avformat_network_init();Report(false);
}
}
}

函数流程:

  • 注册支持的全部编码器(codec)。
  • 注册支持的全部过滤器(filter)。
  • 注册支持的全部封包格式(format)。
  • 初始化支持的网络。
  • 默认关闭打印输出。

4.2 Utils

该模块提供FFMPEG库的工具函数。

4.2.1 Utils定义

namespace media
{
namespace sdk
{
void MEDIASDK_EXPORT Report(bool on);void MEDIASDK_EXPORT Formats();
void MEDIASDK_EXPORT Codecs(bool encoder);
void MEDIASDK_EXPORT Bsfs();
void MEDIASDK_EXPORT Protocols();
void MEDIASDK_EXPORT Filters();
void MEDIASDK_EXPORT PixFmts();
void MEDIASDK_EXPORT SampleFmts();int64_t MEDIASDK_EXPORT Time();
int64_t MEDIASDK_EXPORT NoPtsValue();
int64_t MEDIASDK_EXPORT ToTime(int64_t const& pts, Rational const& timebase);std::vector<std::string> MEDIASDK_EXPORT DisplayCardNames();
}
}

4.2.2 Utils实现

4.2.2.1 Report
void Report(bool on)
{if(on)av_log_set_callback(log_callback_help);		elseav_log_set_callback(log_callback_null);
}

函数说明:

  • 如果需要打印,设置一个可打印新的callback.
  • 如果不要打印,设置一个不可打印的callback.
4.2.2.2 Formats
void Formats()
{AVInputFormat *ifmt  = NULL;AVOutputFormat *ofmt = NULL;const char *last_name;printf("File formats:\n"" D. = Demuxing supported\n"" .E = Muxing supported\n"" --\n");last_name = "000";for (;;) {int decode = 0;int encode = 0;const char *name      = NULL;const char *long_name = NULL;while ((ofmt = (AVOutputFormat *)av_oformat_next(ofmt))) {if ((name == NULL || strcmp(ofmt->name, name) < 0) &&strcmp(ofmt->name, last_name) > 0) {name      = ofmt->name;long_name = ofmt->long_name;encode    = 1;}}while ((ifmt = (AVInputFormat *)av_iformat_next(ifmt))) {if ((name == NULL || strcmp(ifmt->name, name) < 0) &&strcmp(ifmt->name, last_name) > 0) {name      = ifmt->name;long_name = ifmt->long_name;encode    = 0;}if (name && strcmp(ifmt->name, name) == 0)decode = 1;}if (name == NULL)break;last_name = name;printf(" %s%s %-15s %s\n",decode ? "D" : " ",encode ? "E" : " ",name,long_name ? long_name:" ");}
}

函数流程:

  • 遍历输出格式和输入格式
  • 打印名字和长名字
  • 输入格式作为解复用(Demux)打印
  • 输出格式作为复用(Mux)打印
4.2.2.3 Codecs
void Codecs(bool encoder)
{const AVCodecDescriptor **codecs;unsigned i, nb_codecs = get_codecs_sorted(&codecs);printf("%s:\n"" V..... = Video\n"" A..... = Audio\n"" S..... = Subtitle\n"" .F.... = Frame-level multithreading\n"" ..S... = Slice-level multithreading\n"" ...X.. = Codec is experimental\n"" ....B. = Supports draw_horiz_band\n"" .....D = Supports direct rendering method 1\n"" ------\n",encoder ? "Encoders" : "Decoders");for (i = 0; i < nb_codecs; i++) {const AVCodecDescriptor *desc = codecs[i];const AVCodec *codec = NULL;while ((codec = next_codec_for_id(desc->id, codec, encoder))) {printf(" %c", get_media_type_char(desc->type));printf((codec->capabilities & AV_CODEC_CAP_FRAME_THREADS) ? "F" : ".");printf((codec->capabilities & AV_CODEC_CAP_SLICE_THREADS) ? "S" : ".");printf((codec->capabilities & AV_CODEC_CAP_EXPERIMENTAL)  ? "X" : ".");printf((codec->capabilities & AV_CODEC_CAP_DRAW_HORIZ_BAND)?"B" : ".");printf((codec->capabilities & AV_CODEC_CAP_DR1)           ? "D" : ".");printf(" %-20s %s", codec->name, codec->long_name ? codec->long_name : "");if (strcmp(codec->name, desc->name))printf(" (codec %s)", desc->name);printf("\n");}}av_free(codecs);
}

函数流程:

  • 遍历支持的编解码器
  • 按照视频,音频,字幕,帧级多线程,切片级多线程,实验性,支持画横条,支持直接渲染等格式打印
4.2.2.4 Bsfs
void Bsfs()
{const AVBitStreamFilter *bsf = 0;void *opaque = NULL;printf("Bitstream filters:\n");while ((bsf = av_bsf_iterate(&opaque)))printf("%s\n", bsf->name);printf("\n");
}

函数流程:

  • 遍历支持的Bitstream过滤器
  • 打印过滤器名称
4.2.2.5 Protocols
void Protocols()
{void *opaque = NULL;const char *name;printf("Supported file protocols:\n""Input:\n");while ((name = avio_enum_protocols(&opaque, 0)))printf("%s\n", name);printf("Output:\n");while ((name = avio_enum_protocols(&opaque, 1)))printf("%s\n", name);
}

函数流程:

  • 遍历支持的输入/输出协议
  • 打印协议名称
4.2.2.6 Filters
void Filters()
{const AVFilter *filter = NULL;char descr[64], *descr_cur;void *opaque = NULL;int i, j;const AVFilterPad *pad;printf("Filters:\n""  T.. = Timeline support\n""  .S. = Slice threading\n""  ..C = Command support\n""  A = Audio input/output\n""  V = Video input/output\n""  N = Dynamic number and/or type of input/output\n""  | = Source or sink filter\n");while ((filter = av_filter_iterate(&opaque))) {descr_cur = descr;for (i = 0; i < 2; i++) {if (i) {*(descr_cur++) = '-';*(descr_cur++) = '>';}pad = i ? filter->outputs : filter->inputs;for (j = 0; pad && avfilter_pad_get_name(pad, j); j++) {if (descr_cur >= descr + sizeof(descr) - 4)break;*(descr_cur++) = get_media_type_char(avfilter_pad_get_type(pad, j));}if (!j)*(descr_cur++) = ((!i && (filter->flags & AVFILTER_FLAG_DYNAMIC_INPUTS)) ||( i && (filter->flags & AVFILTER_FLAG_DYNAMIC_OUTPUTS))) ? 'N' : '|';}*descr_cur = 0;printf(" %c%c%c %-17s %-10s %s\n",filter->flags & AVFILTER_FLAG_SUPPORT_TIMELINE ? 'T' : '.',filter->flags & AVFILTER_FLAG_SLICE_THREADS    ? 'S' : '.',filter->process_command                        ? 'C' : '.',filter->name, descr, filter->description);}
}

函数流程:

  • 遍历支持的过滤器
  • 按照时间线,切片线程,命令支持,音频输入/输出,视频输入/输出和源/显示类型打印。
4.2.2.7 PixFmts
void PixFmts()
{const AVPixFmtDescriptor *pix_desc = NULL;printf("Pixel formats:\n""I.... = Supported Input  format for conversion\n"".O... = Supported Output format for conversion\n""..H.. = Hardware accelerated format\n""...P. = Paletted format\n""....B = Bitstream format\n""FLAGS NAME            NB_COMPONENTS BITS_PER_PIXEL\n""-----\n");#if !CONFIG_SWSCALE
#   define sws_isSupportedInput(x)  0
#   define sws_isSupportedOutput(x) 0
#endifwhile ((pix_desc = av_pix_fmt_desc_next(pix_desc))) {enum AVPixelFormat av_unused pix_fmt = av_pix_fmt_desc_get_id(pix_desc);printf("%c%c%c%c%c %-16s       %d            %2d\n",sws_isSupportedInput (pix_fmt)              ? 'I' : '.',sws_isSupportedOutput(pix_fmt)              ? 'O' : '.',pix_desc->flags & AV_PIX_FMT_FLAG_HWACCEL   ? 'H' : '.',pix_desc->flags & AV_PIX_FMT_FLAG_PAL       ? 'P' : '.',pix_desc->flags & AV_PIX_FMT_FLAG_BITSTREAM ? 'B' : '.',pix_desc->name,pix_desc->nb_components,av_get_bits_per_pixel(pix_desc));}
}

函数流程:

  • 遍历支持的像素格式
  • 按照输入,输出,硬件支持,调色板和位流类型打印。
4.2.2.8 SampleFmts
void SampleFmts()
{int i;char fmt_str[128];for (i = -1; i < AV_SAMPLE_FMT_NB; i++)printf("%s\n", av_get_sample_fmt_string(fmt_str, sizeof(fmt_str), (enum AVSampleFormat)i));
}

函数流程:

  • 遍历支持的音频采样格式
  • 打印音频采样格式
4.2.2.9 Time
int64_t Time()
{return av_gettime();
}
4.2.2.10 NoPtsValue
int64_t NoPtsValue()
{return AV_NOPTS_VALUE;
}

函数说明:

  • 返回无效时间戳
4.2.2.11 ToTime
int64_t ToTime(int64_t const& pts, Rational const& timebase)
{AVRational tb1 = {timebase.num, timebase.den};AVRational tb2 = {1, AV_TIME_BASE};return av_rescale_q(pts, tb1, tb2);
}

函数说明:

  • 将以1/AV_TIME_BASE秒为单位的时间戳转换位以timebase为单位的时间戳
  • AV_TIME_BASE值为1000000
4.2.2.12 DisplayCardNames
std::vector<std::string> DisplayCardNames()
{std::vector<std::string> displayNames;IDXGIFactory * pFactory;HRESULT hr = CreateDXGIFactory(__uuidof(IDXGIFactory), (void**)(&pFactory));if(FAILED(hr))return displayNames;IDXGIAdapter * pAdapter;int iAdapterNum = 0;while(pFactory->EnumAdapters(iAdapterNum, &pAdapter) != DXGI_ERROR_NOT_FOUND){DXGI_ADAPTER_DESC desc;pAdapter->GetDesc(&desc);std::wstring aa(desc.Description);displayNames.push_back(WStringToString(aa));++iAdapterNum;}return displayNames;
}

函数流程:

  • 遍历系统中显卡名称
  • 将显卡名称放入数组并返回

示例

namespace mediasdk = media::sdk;
int main(int argc, char* argv[])
{mediasdk::Init();mediasdk::Formats();//打印支持的文件格式(封包格式).mediasdk::Codecs(true);//打印支持编码器.mediasdk::Codecs(false);//打印支持的解码器.mediasdk::Protocols();//打印支持的协议.mediasdk::Filters();//打印支持过滤器.mediasdk::PixFmts();//打印支持像素格式.mediasdk::SampleFmts();//打印支持采样格式.std::vector<std::string> names = media::sdk::DisplayCardNames();for(name: names)std::cout << "Display card: " << name << std::endl;return 0;
}

未完待续…

相关文章:

FFMPEG C++封装(二)

4 详细设计 这章是FFMPEG C封装库的详细设计。 4.1 Init 该模块初始化FFMPEG库。 4.1.1 Init定义 namespace media { namespace sdk { void MEDIASDK_EXPORT Init(); } }函数说明&#xff1a; Init 初始化FFMPEG库&#xff0c;该函数可调用多次。 4.1.2 Init实现 name…...

使用unplugin-auto-import页面不引入api飘红

解决方案&#xff1a;. tsconfig.json文件夹加上 {"compilerOptions": {"target": "ES2020","useDefineForClassFields": true,"module": "ESNext","lib": ["ES2020", "DOM", &q…...

八大技术趋势案例(虚拟现实增强现实)

科技巨变,未来已来,八大技术趋势引领数字化时代。信息技术的迅猛发展,深刻改变了我们的生活、工作和生产方式。人工智能、物联网、云计算、大数据、虚拟现实、增强现实、区块链、量子计算等新兴技术在各行各业得到广泛应用,为各个领域带来了新的活力和变革。 为了更好地了解…...

Vue实现SQL语句关键字高亮显示?

SQL关键字高亮 要在Vue中实现SQL语句中关键字的高亮显示&#xff0c;你可以使用类似的方法&#xff0c;但是你需要根据SQL语法的特点来解析并高亮显示关键字。以下是一个示例代码&#xff0c;演示了如何在Vue中实现SQL语句关键字的高亮显示。 <template><div><…...

开始时间大于结束时间

1.dom中代码&#xff0c;监听所选日期值的变化&#xff0c;并把需要比较的时间字段作为参数传到监听方法中&#xff0c; <el-form-item label"起始日期" prop"startTime"><el-date-picker clearable size"small":disabled"isDisa…...

Java中 List 集合,通过 Stream 流进行排序总结

一、数据准备 public class OrderTest {private String channelCode;private BigDecimal rate;// 省略 getter、setter、toString()、constructor }List<OrderTest> orderTestList new ArrayList<>();OrderTest z09 new OrderTest("Z09", new BigDeci…...

1688中国站按关键字搜索工厂数据 API

公共参数 名称类型必须描述keyString是申请免费调用key&#xff08;必须以GET方式拼接在URL中&#xff09;secretString是调用密钥api_nameString是API接口名称&#xff08;包括在请求地址中&#xff09;[item_search,item_get,item_search_shop等]cacheString否[yes,no]默认y…...

YOLOV8逐步分解(2)_DetectionTrainer类初始化过程

接上篇文章yolov8逐步分解(1)--默认参数&超参配置文件加载继续讲解。 1. 默认配置文件加载完成后&#xff0c;创建对象trainer时&#xff0c;需要从默认配置中获取类DetectionTrainer初始化所需的参数args&#xff0c;如下所示 def train(cfgDEFAULT_CFG, use_pythonFalse…...

Java是用什么语言写的?PHP呢?

Java底层是C语言。 Sun公司研发人员根据嵌入式软件的要求&#xff0c;对C进行了改造&#xff0c;去除了留在C的一些不太实用及影响安全的成分&#xff0c;并结合嵌入式系统的实时性要求&#xff0c;开发了一种称为Oak的面向对象语言。而后&#xff0c;经过迭代更新&#xff0c…...

SpringBoot Redis的使用

官方文档&#xff1a; 官方文档&#xff1a;Spring Data Redis :: Spring Data Redis 和jedis一样&#xff0c;SpringBoot Redis 也可以让我在Java代码中使用redis&#xff0c;同样也是通过引入maven依赖的形式。 加速访问github: 使用steam可以免费加速访问github Spring…...

数据仓库——维度表特性

企业信息化工厂 数据集市中的一致性&#xff0c;由于企业信息化工厂的数据集市是从集成仓库中获得信息的&#xff0c;因此至少从维度建模的角度来看&#xff0c;一致性维护的问题减少了。尽管合并不同数据源的问题依然在&#xff0c;但是负担主要在设计者身上。尽管压力降低了…...

从电荷角度理解开关电容中的电荷守恒

目录 一些铺垫电容的电荷量的解释电荷流入流出对节点电压的影响 从电荷角度理解开关电容加法器中的电荷守恒以开关电容积分器为例说明什么样的节点是电荷守恒 一些铺垫 电容的电荷量的解释 对于一个1F的电容&#xff0c;当它的压差为1V时&#xff0c;它所携带的电荷量是QCU1库…...

1.7.1 python 作业 15道

1、求出1 / 1 1 / 3 1 / 5……1 / 99的和 (1分之一1分之三1分支5....) sum0 for i in range(1,100,2): sum 1/i sum; print(sum) 2、用循环语句&#xff0c;计算2 - 10之间整数的循环相乘的值 &#xff08;2*3*4*5....10) sum 1 for i in range(2,11): sum sum *…...

synchronized 和 ReentrantLock 的区别是什么

该文章专注于面试,面试只要回答关键点即可,不需要对框架有非常深入的回答,如果你想应付面试,是足够了,抓住关键点 面试官:synchronized 和 ReentrantLock 的区别是什么 1. 获取锁的方式 synchronized:synchronized的锁获取是隐式的。当线程进入synchronized修饰的代码…...

大话设计模式之迪米特法则

迪米特法则&#xff0c;也称为最少知识原则&#xff08;Law of Demeter&#xff09;&#xff0c;是面向对象设计中的一个重要原则&#xff0c;其核心思想是降低耦合度、减少对象之间的依赖关系&#xff0c;从而使系统更加灵活、易于维护和扩展。 根据迪米特法则&#xff0c;一…...

KSD测试系统使用方法和注意事项

①下载链接在最顶部&#xff1b; ②安装方法&#xff1a;应该先将测试设备绑定在假人身上&#xff0c;测试设备不能过度往下拉&#xff0c;传感器绑在脖子上&#xff0c;切记最后才开传感器开关&#xff01;&#xff01;&#xff01;开传感器后3秒内不要碰测试设备衣服&#x…...

IT服务营销管理案例分析题

习题一 企业随着业务的蓬勃发展&#xff0c;所投入的基础设施资源不断增加。企业员工数倍数增长&#xff0c;办公场地、办公环境等要求也越来越高。 可是该企业的IT部门人员短缺&#xff0c;对IT管理还处于被动的“救火”阶段&#xff0c;每天至少15个突发故障&#xff0c;故障…...

NRF52832修改OTA升级时的bootloader蓝牙MAC

NRF52832在OTA升级时&#xff0c;修改了APP的蓝牙MAC会导致无法升级&#xff0c;原因是OTA程序的蓝牙MAC没有被修改所以手机扫描蓝牙时无法连接 解决办法 在bootloader的程序里面加入修改蓝牙mac地址的代码实现原理&#xff1a; 在bootloader蓝牙广播开启之前修改蓝牙mac 通…...

# Python 编程入门教程

欢迎来到 Python 编程入门教程!Python 是一种简单易学、功能强大的编程语言,适用于各种应用场景,从简单的脚本到大型软件开发项目。无论你是初学者还是有一定编程经验的开发者,本教程都将为你提供全面的学习路径,帮助你掌握 Python 编程的基础知识和技能。 ## 目录 1. 简…...

Sqoop【实践 02】Sqoop1最新版 全库导入 + 数据过滤 + 字段类型支持 说明及举例代码(query参数及字段类型强制转换)

Sqoop1最新版举例 1.环境说明2.import-all-tables3.query4.字段类型支持 1.环境说明 还是之前的环境&#xff1a; # 不必要信息不再贴出 # JDK [roottcloud ~]# java -version java version "1.8.0_251" # MySQL [roottcloud ~]# mysql -V mysql Ver 14.14 Distrib…...

[特殊字符] 智能合约中的数据是如何在区块链中保持一致的?

&#x1f9e0; 智能合约中的数据是如何在区块链中保持一致的&#xff1f; 为什么所有区块链节点都能得出相同结果&#xff1f;合约调用这么复杂&#xff0c;状态真能保持一致吗&#xff1f;本篇带你从底层视角理解“状态一致性”的真相。 一、智能合约的数据存储在哪里&#xf…...

Lombok 的 @Data 注解失效,未生成 getter/setter 方法引发的HTTP 406 错误

HTTP 状态码 406 (Not Acceptable) 和 500 (Internal Server Error) 是两类完全不同的错误&#xff0c;它们的含义、原因和解决方法都有显著区别。以下是详细对比&#xff1a; 1. HTTP 406 (Not Acceptable) 含义&#xff1a; 客户端请求的内容类型与服务器支持的内容类型不匹…...

云计算——弹性云计算器(ECS)

弹性云服务器&#xff1a;ECS 概述 云计算重构了ICT系统&#xff0c;云计算平台厂商推出使得厂家能够主要关注应用管理而非平台管理的云平台&#xff0c;包含如下主要概念。 ECS&#xff08;Elastic Cloud Server&#xff09;&#xff1a;即弹性云服务器&#xff0c;是云计算…...

脑机新手指南(八):OpenBCI_GUI:从环境搭建到数据可视化(下)

一、数据处理与分析实战 &#xff08;一&#xff09;实时滤波与参数调整 基础滤波操作 60Hz 工频滤波&#xff1a;勾选界面右侧 “60Hz” 复选框&#xff0c;可有效抑制电网干扰&#xff08;适用于北美地区&#xff0c;欧洲用户可调整为 50Hz&#xff09;。 平滑处理&…...

ios苹果系统,js 滑动屏幕、锚定无效

现象&#xff1a;window.addEventListener监听touch无效&#xff0c;划不动屏幕&#xff0c;但是代码逻辑都有执行到。 scrollIntoView也无效。 原因&#xff1a;这是因为 iOS 的触摸事件处理机制和 touch-action: none 的设置有关。ios有太多得交互动作&#xff0c;从而会影响…...

selenium学习实战【Python爬虫】

selenium学习实战【Python爬虫】 文章目录 selenium学习实战【Python爬虫】一、声明二、学习目标三、安装依赖3.1 安装selenium库3.2 安装浏览器驱动3.2.1 查看Edge版本3.2.2 驱动安装 四、代码讲解4.1 配置浏览器4.2 加载更多4.3 寻找内容4.4 完整代码 五、报告文件爬取5.1 提…...

佰力博科技与您探讨热释电测量的几种方法

热释电的测量主要涉及热释电系数的测定&#xff0c;这是表征热释电材料性能的重要参数。热释电系数的测量方法主要包括静态法、动态法和积分电荷法。其中&#xff0c;积分电荷法最为常用&#xff0c;其原理是通过测量在电容器上积累的热释电电荷&#xff0c;从而确定热释电系数…...

Python 高效图像帧提取与视频编码:实战指南

Python 高效图像帧提取与视频编码:实战指南 在音视频处理领域,图像帧提取与视频编码是基础但极具挑战性的任务。Python 结合强大的第三方库(如 OpenCV、FFmpeg、PyAV),可以高效处理视频流,实现快速帧提取、压缩编码等关键功能。本文将深入介绍如何优化这些流程,提高处理…...

沙箱虚拟化技术虚拟机容器之间的关系详解

问题 沙箱、虚拟化、容器三者分开一一介绍的话我知道他们各自都是什么东西&#xff0c;但是如果把三者放在一起&#xff0c;它们之间到底什么关系&#xff1f;又有什么联系呢&#xff1f;我不是很明白&#xff01;&#xff01;&#xff01; 就比如说&#xff1a; 沙箱&#…...

npm安装electron下载太慢,导致报错

npm安装electron下载太慢&#xff0c;导致报错 背景 想学习electron框架做个桌面应用&#xff0c;卡在了安装依赖&#xff08;无语了&#xff09;。。。一开始以为node版本或者npm版本太低问题&#xff0c;调整版本后还是报错。偶尔执行install命令后&#xff0c;可以开始下载…...