基于现代 C++17 的模块化视频质量诊断处理流程设计
文章目录
- 0. 引言
- 1. 原始设计分析
- 2. 新的设计思路
- 2.1 定义通用的检测接口
- 2.2 使用 `std::function` 和 `std::any` 管理检测模块
- 2.3 构建可动态配置的检测管道
- 3. 示例实现
- 3.1 定义检测接口和模块
- 3.1.1 检测接口
- 3.1.2 信号检测模块
- 3.1.3 冻结检测模块
- 3.1.4 其他检测模块
- 3.2 构建检测管道
- 3.2.1 初始化管道
- 3.2.2 创建检测模块实例
- 3.2.3 将检测模块添加到管道
- 3.3 执行检测管道
- 3.4 完整代码示例
- 4. 设计优势
- 4.1 灵活性和可扩展性
- 4.2 模块化和可维护性
- 4.3 利用现代 C++ 特性
- 5. 参考资料
0. 引言
在整理视频质量诊断程序模块代码时发现,处理流程通常是固定的,各种检测功能以函数形式顺序调用。这种设计缺乏灵活性,不易于扩展和维护。本文将使用 C++17 的 std::function
和 std::any
,重新设计视频质量诊断处理流程,实现模块化、可灵活扩展的处理管道。
视频质量诊断涉及多个检测模块,如信号检测、冻结检测、遮挡检测等。
详细见:
- C++基于opencv4的视频质量检测
- C++基于opencv的视频质量检测–图像清晰度检测
- C++基于opencv的视频质量检测–画面冻结检测
- C++基于opencv的视频质量检测–图像抖动检测
- C++基于opencv的视频质量检测–遮挡检测
本文将采用面向接口的设计,使用现代 C++ 特性构建一个可动态配置的处理管道。
1. 原始设计分析
原始代码中,各个检测功能以独立的函数实现,处理流程固定:
signalDetect();
freezeframeDetect();
occlusionDetect();
brightnessDetect();
snownoiseDetect();
stripeDetect();
cameramoveDetect();
sharpnessDetect();
colorDetect();
jitterDetect();
这种设计存在以下问题:
- 缺乏灵活性:处理流程固定,无法根据需求动态调整检测模块的顺序或添加新的检测功能。
- 可扩展性差:添加新的检测功能需要修改主流程代码,增加了维护成本。
- 模块耦合度高:各个检测函数之间可能存在隐式依赖,不利于模块化管理。
2. 新的设计思路
流程图
2.1 定义通用的检测接口
借鉴 COM 的接口设计思想,定义一个通用的检测接口 IVideoQualityDetector
:
class IVideoQualityDetector {
public:virtual int Detect(const cv::Mat& img, const cv::Mat& img_jitterRef, const cv::Mat& img_moveRef,const Threshold_t& threshold, DetectDescribe_t* detectDescribe,DetectData_t* detectData) = 0;virtual ~IVideoQualityDetector() = default;
};
每个具体的检测模块都实现该接口,实现其特定的检测逻辑。
2.2 使用 std::function
和 std::any
管理检测模块
std::function
:用于存储各个检测模块的调用方法,便于在管道中按顺序执行。std::any
:用于存储不同类型的检测模块实例,实现类型擦除,支持在运行时动态管理模块。
2.3 构建可动态配置的检测管道
将各个检测模块实例化后,添加到一个检测管道(如 std::vector<std::function<int(...)>>
)中。这样可以根据需求动态调整检测模块的顺序、添加或移除检测模块。
3. 示例实现
3.1 定义检测接口和模块
3.1.1 检测接口
class IVideoQualityDetector {
public:virtual int Detect(const cv::Mat& img, const cv::Mat& img_jitterRef, const cv::Mat& img_moveRef,const Threshold_t& threshold, DetectDescribe_t* detectDescribe,DetectData_t* detectData) = 0;virtual ~IVideoQualityDetector() = default;
};
3.1.2 信号检测模块
class SignalDetector : public IVideoQualityDetector {
public:int Detect(const cv::Mat& img, const cv::Mat&, const cv::Mat&,const Threshold_t& threshold, DetectDescribe_t* detectDescribe,DetectData_t* detectData) override {// 实现信号检测逻辑if (img.empty()) return NO_DETECT;detectData->signalVal = signalDetect(img);if (detectData->signalVal >= threshold.signalThreshold) {detectDescribe->nosignalDescribe = NO_SIGNAL;return SIGNAL_DETECT;} else {detectDescribe->nosignalDescribe = SIGNAL_NORMAL;}return NO_DETECT;}
};
3.1.3 冻结检测模块
class FreezeFrameDetector : public IVideoQualityDetector {
public:int Detect(const cv::Mat& img, const cv::Mat& img_jitterRef, const cv::Mat&,const Threshold_t& threshold, DetectDescribe_t* detectDescribe,DetectData_t* detectData) override {// 实现冻结检测逻辑if (img.empty() || img_jitterRef.empty()) return NO_DETECT;detectData->freezeFrameVal = 1 - freezeFrameDetect(img, img_jitterRef);if (detectData->freezeFrameVal >= threshold.freezeFrameThreshold) {detectDescribe->freezeFrameDescribe = FREEZEFRAME_YES;return FREEZEFRAME_DETECT;} else {detectDescribe->freezeFrameDescribe = FREEZEFRAME_NO;}return NO_DETECT;}
};
3.1.4 其他检测模块
按照上述方式,实现其他检测模块(遮挡检测、亮度检测、雪花噪点检测等),每个模块继承 IVideoQualityDetector
接口,实现其特定的 Detect
方法。
3.2 构建检测管道
3.2.1 初始化管道
std::vector<std::function<int(const cv::Mat&, const cv::Mat&, const cv::Mat&,const Threshold_t&, DetectDescribe_t*, DetectData_t*)>> detectPipeline;
3.2.2 创建检测模块实例
std::any signalDetector = std::make_unique<SignalDetector>();
std::any freezeFrameDetector = std::make_unique<FreezeFrameDetector>();
// ... 创建其他检测模块实例
3.2.3 将检测模块添加到管道
detectPipeline.push_back([&](const cv::Mat& img, const cv::Mat& img_jitterRef, const cv::Mat& img_moveRef,const Threshold_t& threshold, DetectDescribe_t* detectDescribe,DetectData_t* detectData) {return std::any_cast<SignalDetector&>(signalDetector).Detect(img, img_jitterRef, img_moveRef,threshold, detectDescribe, detectData);
});detectPipeline.push_back([&](const cv::Mat& img, const cv::Mat& img_jitterRef, const cv::Mat& img_moveRef,const Threshold_t& threshold, DetectDescribe_t* detectDescribe,DetectData_t* detectData) {return std::any_cast<FreezeFrameDetector&>(freezeFrameDetector).Detect(img, img_jitterRef, img_moveRef,threshold, detectDescribe, detectData);
});
// ... 将其他检测模块添加到管道
3.3 执行检测管道
for (auto& detectFunc : detectPipeline) {detectFunc(img, img_jitterRef, img_moveRef, threshold, detectDescribe, detectData);
}
3.4 完整代码示例
这里仅展示主要结构,完整代码应包括所有检测模块的实现。
#include <iostream>
#include <vector>
#include <functional>
#include <any>
#include <opencv2/opencv.hpp>// 定义相关的常量、结构体和函数(如 Threshold_t、DetectDescribe_t、DetectData_t、signalDetect 等)class IVideoQualityDetector {
public:virtual int Detect(const cv::Mat& img, const cv::Mat& img_jitterRef, const cv::Mat& img_moveRef,const Threshold_t& threshold, DetectDescribe_t* detectDescribe,DetectData_t* detectData) = 0;virtual ~IVideoQualityDetector() = default;
};class SignalDetector : public IVideoQualityDetector {
public:int Detect(const cv::Mat& img, const cv::Mat&, const cv::Mat&,const Threshold_t& threshold, DetectDescribe_t* detectDescribe,DetectData_t* detectData) override {// 信号检测实现}
};// ... 实现其他检测模块int ROCK_QualityDiagnoseFromMat(const cv::Mat img, cv::Mat img_jitterRef, cv::Mat img_moveRef,Threshold_t threshold, DetectDescribe_t* detectDescribe,DetectData_t* detectData, unsigned int flag) {// 初始化阈值和标志位init_threshold(threshold);init_flag(flag);// 构建检测管道std::vector<std::function<int(const cv::Mat&, const cv::Mat&, const cv::Mat&,const Threshold_t&, DetectDescribe_t*, DetectData_t*)>> detectPipeline;std::any signalDetector = std::make_unique<SignalDetector>();// ... 创建其他检测模块实例// 根据标志位添加需要的检测模块if (flag & SIGNAL_DETECT) {detectPipeline.push_back([&](const cv::Mat& img, const cv::Mat& img_jitterRef, const cv::Mat& img_moveRef,const Threshold_t& threshold, DetectDescribe_t* detectDescribe,DetectData_t* detectData) {return std::any_cast<SignalDetector&>(signalDetector).Detect(img, img_jitterRef, img_moveRef,threshold, detectDescribe, detectData);});}// ... 根据其他标志位添加检测模块// 执行检测管道for (auto& detectFunc : detectPipeline) {detectFunc(img, img_jitterRef, img_moveRef, threshold, detectDescribe, detectData);}return 0;
}
4. 设计优势
4.1 灵活性和可扩展性
- 动态配置检测模块:可以根据需要在运行时添加、移除或重新排列检测模块,无需修改主流程代码。
- 支持新功能的扩展:添加新检测功能只需实现
IVideoQualityDetector
接口,并在构建管道时添加即可。
4.2 模块化和可维护性
- 降低模块耦合度:各个检测模块独立实现,遵循统一的接口,模块之间无直接依赖。
- 易于维护和测试:每个检测模块可以独立开发和测试,提高代码质量。
4.3 利用现代 C++ 特性
std::function
:方便地管理和调用函数对象,实现检测管道的灵活组织。std::any
:实现类型擦除,支持在运行时管理不同类型的检测模块实例。
5. 参考资料
- C++17
std::function
文档 - C++17
std::any
文档 - OpenCV 官方文档
相关文章:
基于现代 C++17 的模块化视频质量诊断处理流程设计
文章目录 0. 引言1. 原始设计分析2. 新的设计思路2.1 定义通用的检测接口2.2 使用 std::function 和 std::any 管理检测模块2.3 构建可动态配置的检测管道 3. 示例实现3.1 定义检测接口和模块3.1.1 检测接口3.1.2 信号检测模块3.1.3 冻结检测模块3.1.4 其他检测模块 3.2 构建检…...
高级java每日一道面试题-2024年10月23日-JVM篇-说一下JVM有哪些垃圾回收算法?
如果有遗漏,评论区告诉我进行补充 面试官: 说一下JVM有哪些垃圾回收算法? 我回答: 在 Java 虚拟机 (JVM) 中,垃圾回收 (Garbage Collection, GC) 是一项非常重要的功能,用于自动管理应用程序的内存。JVM 采用多种垃圾回收算法来决定何时以及如何回收…...

高效文本编辑与导航:Vim中的三种基本模式及粘滞位的深度解析
✨✨ 欢迎大家来访Srlua的博文(づ ̄3 ̄)づ╭❤~✨✨ 🌟🌟 欢迎各位亲爱的读者,感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢,在这里我会分享我的知识和经验。&am…...

w005基于Springboot学生心理咨询评估系统
🙊作者简介:拥有多年开发工作经验,分享技术代码帮助学生学习,独立完成自己的项目或者毕业设计。 代码可以私聊博主获取。🌹赠送计算机毕业设计600个选题excel文件,帮助大学选题。赠送开题报告模板ÿ…...

实战-任意文件下载
实战-任意文件下载 1、开局 开局一个弱口令,正常来讲我们一般是弱口令或者sql,或者未授权 那么这次运气比较好,直接弱口令进去了 直接访问看看有没有功能点,正常做测试我们一定要先找功能点 发现一个文件上传点,不…...
PG数据库之视图详解
1. 视图的基本定义 在PostgreSQL(简称pg)数据库中,视图(View)是一种虚拟表,其内容由SQL查询定义。视图并不实际存储数据,而是在每次查询时根据定义的查询语句动态生成结果。视图可以简化复杂的…...

时间序列预测(十五)——有关Python项目框架的实例分析
#1024程序员节|征文# 在之前的学习中,已经对时间序列预测的相关内容有了大致的了解。为了进一步加深理解,并能够将所学知识应用于实际中,我决定找一个完整的Python框架来进行深入学习。经过寻找,我终于找到了一篇非常具…...

ETL、ELT和反向ETL都有什么不同?怎么选择?
数据处理是现代企业中不可或缺的一部分。随着数据量的不断增长,如何高效地处理、转换和加载数据变得尤为重要。本文将介绍三种常见的数据处理方式:ETL、ELT和反向ETL,帮助读者更好地理解和选择适合自己业务需求的方式。 一、ETL 定义&#…...
linux 中文实用型手册 基于RHEL(红帽系)
硬件系统 Updated by wangjing on 2024-10-28 at 02:36:57 in Tongzhou District, Beijing. 硬件信息 机器型号 dmidecode | grep "Product Name"CPU型号 cat /proc/cpuinfo |grep "model name" | uniqWWWCPU详情 lscpuCPU个数 cat /proc/cpuinfo |grep &q…...

Hash表算法
哈希表 理论知识(本文来自于代码随想录摘抄)什么是哈希常见的三种哈希结数组:set:map:其他常用方法或者技巧(自己总结的) 练习题和讲解有效的字母移位词349. 两个数组的交集1. 两数之和454. 四数相加 II15. 三数之和 总…...

MySQL企业常见架构与调优经验分享
文章目录 一、选择 PerconaServer、MariaDB 还是 MYSQL二、常用的 MYSQL 调优策略三、MYSOL 常见的应用架构分享四、MYSOL 经典应用架构 观看学习课程的笔记,分享于此~ 课程:MySQL企业常见架构与调优经验分享 mysql官方优化文档 调优MySQL参数 一、选择 …...

C++引用类型变量
引用变量的主要用途是用作函数的形参。这样函数将使用原始数据,而不是副本。除指针之外,引用也为处理大型结构提供了一种非常方便的途径。 再C中使用&符号标识引用。也就是说C给&符号赋予了另一个含义,将其用来声明引用。 引用的声…...
《C++23 新特性:现代软件开发的变革力量》
在软件开发的快速演进中,C作为一种强大且广泛应用的编程语言,不断推陈出新以适应日益复杂的开发需求。C23 的到来,为现代软件开发带来了诸多新的机遇和挑战。它的新特性不仅影响着开发者的编程习惯,也在代码效率、可维护性以及软件…...
Educational Codeforces Round 88 E. Modular Stability
题目链接 Educational Codeforces Round 88 E. Modular Stability 思路 对于任意的非负整数 x x x,我们要满足 x % a % b x % b % a x \% a \% b x \% b \% a x%a%bx%b%a。因为 a < b a < b a<b,所以只有 b b b为 a a a的倍数时才满足条件…...

Android中SurfaceView与GLSurfaceView 的关系
SurfaceView 与 GLSurfaceView 的关系 在 Android 开发中,SurfaceView 和 GLSurfaceView 是实现自定义渲染效果的关键组件。它们提供了不同的渲染方式,适用于不同的应用场景。我们将通过以下几个方面详细说明 SurfaceView 和 GLSurfaceView 的特点及实现…...

numpy——数学运算
一、标量——矢量 import numpy as npa 3.14 b np.array([[9, 5], [2, 7]])print(a) print(b)# ---------- 四则运算 ---------- print(a b) # np.add print(a - b) # np.subtract print(a * b) # np.multiply print(a / b) # np.divide 二、矢量——矢量 import nump…...

【工具】Charles对360浏览器抓包抓包
Charles 和 switchy sharp 配合,可以对 Chrome 进行抓包也可以配合对360安全浏览器抓包。 本文以Windows 电脑中的配置为例,介绍如何实现抓包。(Mac中操作基本一致) 1.安装Charles 可根据自己的电脑下载对应的版本:…...

【HarmonyOS】判断应用是否已安装
【HarmonyOS】判断应用是否已安装 前言 在鸿蒙中判断应用是否已安全,只是通过包名是无法判断应用安装与否。在鸿蒙里新增了一种判断应用安装的工具方法,即:canOpenLink。 使用该工具函数的前提是,本应用配置了查询标签querySch…...

Qt Designer客户端安装和插件集(pyqt5和pyside2)
GitHub - PyQt5/QtDesignerPlugins: Qt Designer PluginsQt Designer Plugins. Contribute to PyQt5/QtDesignerPlugins development by creating an account on GitHub.https://github.com/PyQt5/QtDesignerPlugins 一、下载客户端 https://github.com/PyQt5/QtDesigner/rel…...
基于边缘计算的智能门禁系统架构设计分析
案例 阅读以下关于 Web 系统架构设计的叙述,回答问题1至问题3。 【说明】 某公司拟开发一套基于边缘计算的智能门禁系统,用于如园区、新零售、工业现场等存在来访被访业务的场景。来访者在来访前,可以通过线上提前预约的方式将自己的个人信息…...
【Linux】shell脚本忽略错误继续执行
在 shell 脚本中,可以使用 set -e 命令来设置脚本在遇到错误时退出执行。如果你希望脚本忽略错误并继续执行,可以在脚本开头添加 set e 命令来取消该设置。 举例1 #!/bin/bash# 取消 set -e 的设置 set e# 执行命令,并忽略错误 rm somefile…...

CTF show Web 红包题第六弹
提示 1.不是SQL注入 2.需要找关键源码 思路 进入页面发现是一个登录框,很难让人不联想到SQL注入,但提示都说了不是SQL注入,所以就不往这方面想了 先查看一下网页源码,发现一段JavaScript代码,有一个关键类ctfs…...
Java 8 Stream API 入门到实践详解
一、告别 for 循环! 传统痛点: Java 8 之前,集合操作离不开冗长的 for 循环和匿名类。例如,过滤列表中的偶数: List<Integer> list Arrays.asList(1, 2, 3, 4, 5); List<Integer> evens new ArrayList…...

练习(含atoi的模拟实现,自定义类型等练习)
一、结构体大小的计算及位段 (结构体大小计算及位段 详解请看:自定义类型:结构体进阶-CSDN博客) 1.在32位系统环境,编译选项为4字节对齐,那么sizeof(A)和sizeof(B)是多少? #pragma pack(4)st…...
Neo4j 集群管理:原理、技术与最佳实践深度解析
Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...
.Net Framework 4/C# 关键字(非常用,持续更新...)
一、is 关键字 is 关键字用于检查对象是否于给定类型兼容,如果兼容将返回 true,如果不兼容则返回 false,在进行类型转换前,可以先使用 is 关键字判断对象是否与指定类型兼容,如果兼容才进行转换,这样的转换是安全的。 例如有:首先创建一个字符串对象,然后将字符串对象隐…...
离线语音识别方案分析
随着人工智能技术的不断发展,语音识别技术也得到了广泛的应用,从智能家居到车载系统,语音识别正在改变我们与设备的交互方式。尤其是离线语音识别,由于其在没有网络连接的情况下仍然能提供稳定、准确的语音处理能力,广…...

算术操作符与类型转换:从基础到精通
目录 前言:从基础到实践——探索运算符与类型转换的奥秘 算术操作符超级详解 算术操作符:、-、*、/、% 赋值操作符:和复合赋值 单⽬操作符:、--、、- 前言:从基础到实践——探索运算符与类型转换的奥秘 在先前的文…...
LangChain 中的文档加载器(Loader)与文本切分器(Splitter)详解《二》
🧠 LangChain 中 TextSplitter 的使用详解:从基础到进阶(附代码) 一、前言 在处理大规模文本数据时,特别是在构建知识库或进行大模型训练与推理时,文本切分(Text Splitting) 是一个…...
字符串哈希+KMP
P10468 兔子与兔子 #include<bits/stdc.h> using namespace std; typedef unsigned long long ull; const int N 1000010; ull a[N], pw[N]; int n; ull gethash(int l, int r){return a[r] - a[l - 1] * pw[r - l 1]; } signed main(){ios::sync_with_stdio(false), …...