【Qt】spdlog日志模块的使用
版本
spdlog版本:1.5.0
采用1.5.0版本主要基于以下考虑:兼容Qt5.9.X版本和兼容C++11。
spdlog 1.5.0下载地址:https://github.com/gabime/spdlog/releases/tag/v1.5.0
摘要
在Qt应用程序开发中,良好的日志系统至关重要。本文将介绍如何使用spdlog 1.5.0创建满足以下要求的日志系统:
-
自定义文件名格式:yyyyMMdd_hhmmss_毫秒.log,不使用spdlog提供的日志轮转功能
spdlog::sinks::rotating_file_sink_mt,采用自定义custom_rotating_file_sink;
-
保留最近10个日志文件,每个日志文件大小限制为1MB。
例子
logmanager.h文件
#ifndef LOGMANAGER_H
#define LOGMANAGER_H#include <QObject>
#include <memory>
#include <spdlog/spdlog.h>class LogManager : public QObject
{Q_OBJECT
public:static LogManager& instance();void initialize(const QString& logDir = "logs",const QString& appName = "app",size_t maxFileSize = 1024 * 1024, // 1MBsize_t maxFiles = 10);void shutdown();template<typename... Args>static void log(spdlog::level::level_enum level, const QString& message, Args... args){if (instance().m_logger){instance().m_logger->log(level, message.toStdString().c_str(), args...);}}// 便捷方法static void trace(const QString& message){log(spdlog::level::trace, message);}static void debug(const QString& message){log(spdlog::level::debug, message);}static void info(const QString& message){log(spdlog::level::info, message);}static void warn(const QString& message){log(spdlog::level::warn, message);}static void error(const QString& message){log(spdlog::level::err, message);}static void critical(const QString& message){log(spdlog::level::critical, message);}private:LogManager(QObject* parent = nullptr);~LogManager();std::shared_ptr<spdlog::logger> createCustomLogger(const std::string& base_filename,size_t max_size,size_t max_files);std::shared_ptr<spdlog::logger> m_logger;std::atomic<bool> m_shuttingDown{false};signals:void aboutToShutdown();private slots:void onAboutToQuit();};// 日志宏定义
#define LOG_TRACE(...) LogManager::log(spdlog::level::trace, __VA_ARGS__)
#define LOG_DEBUG(...) LogManager::log(spdlog::level::debug, __VA_ARGS__)
#define LOG_INFO(...) LogManager::log(spdlog::level::info, __VA_ARGS__)
#define LOG_WARN(...) LogManager::log(spdlog::level::warn, __VA_ARGS__)
#define LOG_ERROR(...) LogManager::log(spdlog::level::err, __VA_ARGS__)
#define LOG_CRITICAL(...) LogManager::log(spdlog::level::critical, __VA_ARGS__)#endif // LOGMANAGER_H
logmanager.cpp文件
#include "logmanager.h"
#include <spdlog/sinks/base_sink.h>
#include <spdlog/details/file_helper.h>
#include <mutex>
#include <chrono>
#include <iomanip>
#include <sstream>
#include <vector>
#include <algorithm>
#include <QDir>
#include <QFileInfo>
#include <QDateTime>
#include <QCoreApplication>
#include <csignal>
#include <QDebug>// 替换 std::filesystem 的 C++11 兼容实现
namespace spdlog
{class custom_rotating_file_sink : public spdlog::sinks::base_sink<std::mutex>{public:custom_rotating_file_sink(const std::string& base_filename,std::size_t max_size,std::size_t max_files): base_filename_(base_filename),max_size_(max_size),max_files_(max_files){file_helper_.open(gen_filename());}protected:void sink_it_(const spdlog::details::log_msg& msg) override{spdlog::memory_buf_t formatted;formatter_->format(msg, formatted);if (file_helper_.size() + formatted.size() > max_size_){rotate_();}file_helper_.write(formatted);}void flush_() override{file_helper_.flush();}private:std::string gen_filename(){QDateTime now = QDateTime::currentDateTime();QString timeStr = now.toString("yyyyMMddhhmmss");// 添加毫秒部分(3位)int ms = now.time().msec();timeStr += QString("_%1").arg(ms, 3, 10, QLatin1Char('0'));return base_filename_ + "_" + timeStr.toStdString() + ".log";}void rotate_(){file_helper_.close();cleanup_old_files();file_helper_.open(gen_filename());}void cleanup_old_files(){if (max_files_ == 0) return;QFileInfo base_info(QString::fromStdString(base_filename_));QDir dir = base_info.absoluteDir();QString base_name = base_info.fileName();QFileInfoList files = dir.entryInfoList(QStringList() << (base_name + "_*.log"),QDir::Files, QDir::Time);// 删除最旧的文件while (files.size() >= static_cast<int>(max_files_)){QFile::remove(files.last().absoluteFilePath());files.removeLast();}}std::string base_filename_;std::size_t max_size_;std::size_t max_files_;spdlog::details::file_helper file_helper_;};} // namespaceLogManager::LogManager(QObject* parent) : QObject(parent)
{// 连接Qt退出信号connect(qApp, &QCoreApplication::aboutToQuit, this, &LogManager::onAboutToQuit);// 处理异常信号static auto handleSignal = [](int){LogManager::instance().shutdown();std::_Exit(1);};std::signal(SIGTERM, handleSignal);std::signal(SIGSEGV, handleSignal);std::signal(SIGINT, handleSignal);std::signal(SIGABRT, handleSignal);
}LogManager::~LogManager()
{shutdown();
}LogManager& LogManager::instance()
{static LogManager instance;return instance;
}void LogManager::initialize(const QString& logDir, const QString& appName, size_t maxFileSize, size_t maxFiles)
{if (m_logger){return;}// 确保日志目录存在QDir().mkpath(logDir);std::string base_filename = QDir(logDir).absoluteFilePath(appName).toStdString();m_logger = createCustomLogger(base_filename, maxFileSize, maxFiles);// 设置默认日志格式m_logger->set_pattern("[%Y-%m-%d %H:%M:%S.%e] [%l] [thread %t] %v");m_logger->set_level(spdlog::level::trace);spdlog::register_logger(m_logger);spdlog::set_default_logger(m_logger);
}void LogManager::shutdown()
{/*if (m_logger){spdlog::drop(m_logger->name());m_logger.reset();}spdlog::shutdown();*/if (m_shuttingDown) return;m_shuttingDown = true;emit aboutToShutdown();try{if (m_logger){m_logger->flush();spdlog::drop(m_logger->name());}spdlog::shutdown();m_logger.reset();}catch (const spdlog::spdlog_ex& ex){qCritical() << "Log shutdown error:" << ex.what();}
}void LogManager::onAboutToQuit()
{shutdown();
}std::shared_ptr<spdlog::logger> LogManager::createCustomLogger(const std::string& base_filename,size_t max_size,size_t max_files)
{auto sink = std::make_shared<spdlog::custom_rotating_file_sink>(base_filename, max_size, max_files);auto logger = std::make_shared<spdlog::logger>("qt_logger", sink);return logger;
}
main.cpp文件
#include <QCoreApplication>
#include "logmanager.h"
#include <QTimer>
#include <QDebug>int main(int argc, char* argv[])
{QCoreApplication a(argc, argv);// 初始化日志系统LogManager::instance().initialize("logs", "MyAppTest");// 连接关闭信号进行额外清理QObject::connect(&LogManager::instance(), &LogManager::aboutToShutdown, [](){LOG_INFO("Performing final cleanup before shutdown...");});for (int i = 0; i < 5000; ++i){LOG_INFO("This is a test message to fill up the log file. Iteration: {}", i);}return a.exec();
}
相关文章:
【Qt】spdlog日志模块的使用
版本 spdlog版本:1.5.0 采用1.5.0版本主要基于以下考虑:兼容Qt5.9.X版本和兼容C11。 spdlog 1.5.0下载地址:https://github.com/gabime/spdlog/releases/tag/v1.5.0 摘要 在Qt应用程序开发中,良好的日志系统至关重要。本文将介…...
python相关面试题
python相关面试题 1.上下文管理器需要实现哪两种方法以及相关应用 2.对比一下进程,线程和协程 3.魔法函数有哪些 4.什么是闭包和装饰器,装饰器缺点是什么 5.什么是浅拷贝和深拷贝 6.什么是GIL锁和互斥锁 7.init和new有什么区别,new方法的返回…...
swift菜鸟教程11-12(数组与字典)
一个朴实无华的目录 今日学习内容:1.Swift 数组1.1创建数组1.2访问数组1.3修改数组使用 append() 方法或者赋值运算符 在数组末尾添加元素通过索引修改数组元素的值: 1.4遍历数组 使用for-in循环同时需要每个数据项的值和索引值 1.5合并数组1.6count 属…...
.NET WPF 控件类分层结构
.NET WPF 控件类分层结构 在 WPF 中,类的层级结构设计是为了实现线程安全、依赖属性、可视化渲染、布局和交互等功能。以下是 WPF 核心基类的逐级说明,从最底层到最顶层: 1 DispatcherObject 作用: 提供与 WPF 线程模型…...
基于ImGui+FFmpeg实现播放器
基于ImGuiFFmpeg实现播放器 演示: ImGui播放器 继续研究FFmpeg,之前做了一个SDL的播放器,发现SDL的可视化UI界面的功能稍微差了点,所以今天我们换了一个新的工具,也就是ImGui。 ImGui官方文档:https://g…...
【Docker】快速部署 Certbot 并为 Nginx 服务器配置 SSL/TLS 证书
【Docker】快速部署 Certbot 并为 Nginx 服务器配置 SSL/TLS 证书 引言 Certbot 是一个免费的开源工具,用于自动化管理和获取 SSL/TLS 证书,主要用于与 Let’s Encrypt 证书颁发机构交互。 步骤 Nginx 挂载 certbot 文件夹。 docker run -d \--name…...
MATLAB编写的机械臂控制仿真程序,它主要实现了对一个二连杆机械臂的运动控制仿真,比较了PID控制和非线性模型预测控制两种方法在机械臂轨迹跟踪任务中的性能
clc; clear; close all;%% 机械臂参数 l1 0.5; l2 0.4; Ts 0.02; sim_time 60; t 0:Ts:sim_time;%% 物理参数 m1 1.0; m2 0.8; g 9.81;%% 直线轨迹参数 start_point [0.3; 0.1]; end_point [0.7; 0.3]; progress t/sim_time; xd start_point(1) (end_point(1) - s…...
python办公自动化------word文件的操作
一、 word文件的创建 需要安装包:python-docx 例1:创建word文件 from docx import Document# 创建doc对象 doc Document()# 指定路径下创建一个docx文件 doc.save(./dataFile/test1_doc.docx) 运行结果: 例2:word中添加内容 …...
Python小程序 - 文件处理3:正则表达式
正则表达式:文本年鉴表。遗留的问题很多。。。用AI再想想 需求:读入txt文件,过滤文件有关年记录 0)读入txt文件 1)以“。”,中文句号,为界区分一句,最小统计单位 2)年格…...
JAVA中正则表达式的入门与使用
JAVA中正则表达式的入门与使用 一,基础概念 正则表达式(Regex) 用于匹配字符串中的特定模式,Java 中通过 java.util.regex 包实现,核心类为: Pattern:编译后的正则表达式对象。 Matcher&#…...
智慧能源管理平台:驱动电网数字化转型,引领绿色能源新时代
安科瑞 华楠18706163979 一、引言 在全球能源转型和"双碳"目标的推动下,微电网作为分布式能源系统的重要组成部分,正迎来快速发展期。安科瑞电气股份有限公司推出的微电网智慧能源管理平台(EMS 3.0),通过整…...
OpenCV 图形API(30)图像滤波-----腐蚀操作函数erode()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 使用特定的结构元素腐蚀图像。 cv::gapi::erode 是 OpenCV 的 G-API 模块中用于执行图像腐蚀操作的函数。腐蚀是一种基本的形态学操作࿰…...
02-MySQL 面试题-mk
1.如何定位慢查询? 什么是慢查询? 页面加载过慢、接口压测响应时间过长(超过1s) 慢查询出现的情况有哪些? 聚合查询多表查询表数据量过大查询深度分页查询如何定位慢查询? 方案一:开源工具 调试工具:Arthas运维工具:Prometheus、Skywalking**Arthas:**可以使用命令的…...
利用安固软件实现电脑屏幕录像:四种实用方法分享
在日常工作中,有时我们需要录制电脑屏幕以进行教学演示、软件操作记录或重要会议的存档。安固软件终端安全管理系统提供了强大的屏幕录像功能,可以满足这些需求。 接下来,本文将介绍如何使用安固软件设置电脑实时画面录像,并分享…...
Gitee DevSecOps 以 CBB 驱动军工研发范式革新:平台化管理构件实践
随着军工软件向智能化、标准化与集约化发展,传统研发模式在效率、质量及协同方面面临显著瓶颈。项目规模扩大与系统复杂度的提升,亟需一种创新研发范式。Gitee DevSecOps 平台基于 CBB(通用构件库)理念,通过模块化、标…...
spring:xml方式调用构造方法创建Bean,调用set方法配置字段
如题: 要创建的Bean类UserServiceImpl02 : package com.itheima.service.impl;import com.itheima.dao.interfaces.InterfaceUserDao; import com.itheima.service.interfaces.InterfaceUserService;/*** copyright 2003-2024* author qiao wei* da…...
PM2 完全指南:Node.js 应用后台启动、关闭与重启详解
文章目录 **PM2 完全指南:Node.js 应用后台启动、关闭与重启详解****1. 什么是 PM2?****2. 安装 PM2****全局安装****验证安装** **3. 使用 PM2 启动 Node.js 应用****基本启动****指定应用名称****集群模式(多进程负载均衡)****监…...
LeetCode算法题(Go语言实现)_40
题目 给你一个二叉树的根节点 root。设根节点位于二叉树的第 1 层,而根节点的子节点位于第 2 层,依此类推。 请返回层内元素之和 最大 的那几层(可能只有一层)的层号,并返回其中 最小 的那个。 一、代码实现 import &…...
网关与路由器知识点
1.自治系统是由同构型的网关连接的因特网。自治系统内部的网关之间执行内部网关协议IGP,半天自治系统之间用外部网关协议EGP交换路由信息。 2.BGP4是一种动态路由发现协议,支持无类别域间路由CIDR,应用于不同ISP的网络之间,成为事…...
【Hadoop入门】Hadoop生态之Pig简介
1 什么是Pig? 在大数据分析领域,Apache Pig是一个不可忽视的重要工具。Pig是Apache Hadoop生态系统中的一个高级数据分析平台,它提供了一种称为Pig Latin的脚本语言,用于简化大规模数据集的并行处理。Pig的核心思想是将复杂的数据…...
TPS入门DAY04 服务器篇
1.创建委托并绑定回调,实现解耦 /* 创建会话完成 */ DECLARE_DYNAMIC_MULTICAST_DELEGATE_OneParam(FMultiPlayerOnCreateSessionCompleted, bool, bWasSuccessful); /* 开始会话完成 */ DECLARE_DYNAMIC_MULTICAST_DELEGATE_OneParam(FMultiPlayerOnStartSessionC…...
YOLO学习笔记 | YOLOv8环境搭建全流程指南(2025.4)
===================================================== github:https://github.com/MichaelBeechan CSDN:https://blog.csdn.net/u011344545 ===================================================== YOLOv8环境搭建 一、环境准备与工具配置1. Conda虚拟环境搭建2. CUDA与…...
Java 设计模式:适配器模式详解
Java 设计模式:适配器模式详解 适配器模式(Adapter Pattern)是一种结构型设计模式,它通过将一个类的接口转换为客户端期望的另一个接口,使原本不兼容的类能够协同工作。适配器模式就像现实生活中的电源适配器…...
Java的基本语法(1)
一、运算符和表达式 举例说明什么是运算符,什么是表达式: int a 1; int b 2; int c a b; 在这个例子当中,是运算符,并且是算术运算符 ab是表达式,因为是运算符,所以ab是算术表达式 1.1算术运算符 …...
Spark-Core编程二
23) sortByKey import org.apache.spark.rdd.RDD import org.apache.spark.{SparkConf, SparkContext} object Cww { def main(args: Array[String]): Unit { // 创建 SparkConf 对象,设置应用名称和运行模式 val conf new SparkConf().setAppName("SortBy…...
【antd + vue】Modal 对话框:修改弹窗标题样式、Modal.confirm自定义使用
一、标题样式 1、目标样式:修改弹窗标题样式 2、问题: 直接在对应css文件中修改样式不生效。 3、原因分析: 可能原因: 选择器权重不够,把在控制台找到的选择器直接复制下来,如果还不够就再加ÿ…...
JavaWeb 课堂笔记 —— 08 请求响应
本系列为笔者学习JavaWeb的课堂笔记,视频资源为B站黑马程序员出品的《黑马程序员JavaWeb开发教程,实现javaweb企业开发全流程(涵盖SpringMyBatisSpringMVCSpringBoot等)》,章节分布参考视频教程,为同样学习…...
DataWorks智能体Agent发布!基于MCP实现数据开发与治理自动化运行
在传统的数据开发工作中,企业用户或者开发者常常需要进行繁琐的配置、复杂的代码撰写、反复的性能调优和大量重复性的操作,才能实现数据开发、数据集成和数据治理等工作,效率十分低下。 近日,阿里云大数据开发治理平台DataWorks基…...
VitePress 项目部署 cloudflare page 提示 Node 构建错误
提示的构建错误信息为: 09:35:29.838 Error: Exit with error code: 1 09:35:29.839 at ChildProcess.<anonymous> (/snapshot/dist/run-build.js) 09:35:29.839 at Object.onceWrapper (node:events:652:26) 09:35:29.839 at ChildProcess.emit …...
VUE的node包缓存很严重,问题及解决办法
以下是一些可能导致缓存问题的地方以及如何检查和解决这些问题的建议: 1. 检查依赖包的版本是否过时 某些依赖包可能有已知的缓存问题或性能问题。以下是需要特别注意的几个包: Vue CLI 相关: vue/cli-service 和 vue/cli-plugin-babel 的版…...
<C#>在 .NET 开发中,依赖注入, 注册一个接口的多个实现
在 .NET 开发里,有时一个接口会有多个实现类,此时就需要向依赖注入容器注册多个实现。下面会详细介绍不同场景下如何注册多个实现,以及怎样从容器中解析这些实现。 1. 注册多个实现 在 .NET 中,依赖注入容器可以通过不同方式注册…...
计算机视觉色彩空间全解析:RGB、HSV与Lab的实战对比
计算机视觉色彩空间全解析:RGB、HSV与Lab的实战对比 一、前言二、RGB 色彩空间2.1 RGB 色彩空间原理2.1.1 基本概念2.1.2 颜色混合机制 2.2 RGB 在计算机视觉中的应用2.2.1 图像读取与显示2.2.2 颜色识别2.2.3 RGB 色彩空间的局限性 三、HSV 色彩空…...
美食推荐小程序
用户端 1. **定位与筛选** - **精准定位**:根据用户所在位置,推荐周边美食餐厅。 - **筛选条件**:支持按菜系(如川菜、粤菜)、价格区间、评分、距离远近等筛选餐厅。 2. **餐厅展示** - **基本信息**:…...
使用django实现windows任务调度管理
在 Django 中实现 Windows 任务调度管理,你可以使用几种不同的方法。最常见的方法是使用 Django 自带的 celery 或者 django-background-tasks 库,或者使用 Windows 自带的任务计划程序。下面我会分别介绍这几种方法: 方法 1:使用…...
【Android】Android 打包 Release 崩溃问题全解析:Lint 错误、混淆类丢失及解决方法大全
摘要: 在 Android 项目的 Release 打包过程中,经常遇到诸如 Lint 校验失败、程序闪退、类找不到等问题。本文将详细分析 Android 打包时常见的崩溃原因,特别是如何应对 Lint 报错、混淆引发的类丢失(NoClassDefFoundError…...
【Python 开源】你的 Windows 关机助手——PyQt5 版定时关机工具
🖥️ 你的 Windows 关机助手——PyQt5 版定时关机工具 相关资源文件已经打包成EXE文件,可双击直接运行程序,且文章末尾已附上相关源码,以供大家学习交流,博主主页还有更多Python相关程序案例,秉着开源精神的…...
布局决定终局:基于开源AI大模型、AI智能名片与S2B2C商城小程序的战略反推思维
摘要:在商业竞争日益激烈的当下,布局与终局预判成为企业成功的关键要素。本文探讨了布局与终局预判的智慧性,强调其虽无法做到百分之百准确,但能显著提升思考能力。终局思维作为重要战略工具,并非一步到位的战略部署&a…...
vue3循环表单【以el-form组件为例】,如何校验所有表单,所有表单校验通过后提交
1.循环时,在组件标签上使用ref绑定组件实例 3.如何校验所有表单都通过后方可提交?利用promise.all()判断 代码如下: <template><div><!-- 循环渲染表单 --><el-formv-for"(formItem, index) in formList":…...
量子代理签名:量子时代的数字授权革命
1. 量子代理签名的定义与核心原理 量子代理签名(Quantum Proxy Signature, QPS)是经典代理签名在量子信息领域的延伸,允许原始签名者(Original Signer)授权给代理签名者(Proxy Signer)代为签署文…...
HTTPS代理的实际优势,如何选择HTTPS代理服务?
在数字化时代,网络请求的安全性和效率成为企业和个人用户关注的焦点。HTTPS代理作为一种强大的工具,不仅在数据安全方面表现出色,还在安全访问、突破地域限制以及支持复杂任务等方面展现出多样化的优势。本文将深入探讨HTTPS代理的实际优势&a…...
Java 中常见的数据结构
目录 1. List (列表) 2)ArrayList 2)LinkedList 2. Set (集合) 1)HashSet 2)TreeSet 3. Map (映射) 1)HashMap 2)TreeMap 4. Queue (队列) 1)LinkedList (也实现了Queue接口) 2&…...
Transformer多卡训练初始化分布式环境:(backend=‘nccl‘)
Transformer多卡训练初始化分布式环境:(backend=‘nccl’) dist.init_process_group(backend=nccl)在多卡环境下初始化分布式训练环境,并为每个进程分配对应的 GPU 设备。下面为你逐行解释代码的含义: 1. 初始化分布式进程组 try:dist.init_process_group(backend=nccl) e…...
云曦月末断网考核复现
Web 先看一个BUUCTF中的文件一个上传题 [BUUCTF] 2020新生赛 Upload 打开后是一个文件上传页面 随便上传一个txt一句话木马后出现js弹窗,提示只能上传图片格式文件 说明有前端验证。我的做法是把一句话改为.jpg格式, 然后上传 访问发现虽然上传成功了…...
SQL Server AlwaysOn (SQL 查询数据详解及监控用途)
修正后的完整查询 SELECT ar.replica_server_name AS [副本名称],ar.availability_mode_desc AS [同步模式],DB_NAME(dbr.database_id) AS [数据库名称],dbr.database_state_desc AS [数据库状态],dbr.synchronization_state_desc AS [同步状态],dbr.synchronization_health_d…...
使用 Q - learning 算法解决迷宫路径规划问题
整体功能概述 这段 Python 代码实现了一个使用 Q - learning 算法解决迷宫路径规划问题的程序。智能体在给定的迷宫环境中学习如何找到从起点到终点的最优路径,以获得最大奖励。 模块导入 python import numpy as np import randomnumpy:用于处理数组…...
SqlYog无限试用方法
1、WinR ,输入 :regedit 回车 2、进入注册表,在 \HEYK_CURRENT_USER\Software\{*********-D8CE-4637-9BC7-93E************}下的【InD100】保存着SQLyog的使用天数 3、在【InD】上右键,点击删除该项,在重启SQLyog后注册表中会重…...
14 nginx 的 dns 缓存的流程
前言 这个是 2020年11月 记录的这个关于 nginx 的 dns 缓存的问题 docker 环境下面 前端A连到后端B 前端B连到后端A 最近从草稿箱发布这个问题的时候, 重新看了一下 发现该问题的记录中仅仅是 定位到了 nginx 这边的 dns 缓存的问题, 但是 并没有到细节, 没有到 具体的 n种…...
Spring Cloud Gateway 具体的实现案例
文章目录 前言✅ 1. **创建 Spring Boot 项目**Maven 依赖:Gradle 依赖: ✅ 2. **配置 application.yml 路由和过滤器**解释: ✅ 3. **创建自定义过滤器**3.1 **前置过滤器(Pre Filter)**3.2 **后置过滤器(…...
css易混淆的知识点
子选择器 (>) vs 后代选择器 (空格) 子选择器 (>) 只匹配直接子元素。后代选择器 (空格) 匹配所有后代元素(无论嵌套多深)。 绝对定位vs相对定位 布局: justify-content 的作用 控制子元素在主轴上的分布方式。常见值包括 flex-start、…...
OpenCV 图形API(27)图像滤波-----膨胀函数dilate()
操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C11 算法描述 使用特定的结构元素膨胀图像。 cv::gapi::dilate 是 OpenCV G-API 模块中的一个函数,用于对图像执行膨胀操作。膨胀是一种形态学操作…...