news 2026/2/27 19:21:05

可分离滤波-高斯滤波

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
可分离滤波-高斯滤波

一、概述

采用一维高斯滤波(可分离高斯滤波)的核心原因在于:在保持与二维高斯滤波完全相同数学效果的前提下,大幅降低计算复杂度并提升工程效率。由于二维高斯核可以严格分解为两个相互独立的一维高斯核(横向 × 纵向),可大大降低计算量,对于大小为M×n的图像和大小为m×n的核,实现卷积操作需要MNmn次乘法核加分运算(计算公式如下3.35),但是如果滤波核是可分离的,对于w1(行)卷积核,第一次卷积只需要MNm次乘法和加分运算,因为w1卷积核的大小为m×1,对于w2(列)卷积核,第二次卷积只需要MNn次乘法和加分运算,所以共需要MN(m+n)次乘法和加分运算,因此可分离的核执行卷积运算可以大大减少计算量,滤波核越大越明显(可见如下公式3.44)。

二、二维高斯滤波

首先我们讲解一下二维高斯滤波核的生成,这里根据高斯函数可直接生成高斯滤波核,代码如下:

// 二维高斯滤波核 std::vector<std::vector<double>> GaussianKernelGenerate(int kSize, double sigma) { int k = kSize / 2; std::vector<std::vector<double>> kernel(kSize, std::vector<double>(kSize)); double sum = 0.0; const double PI = 3.14159265358979323846; for (int i = -k; i <= k; ++i) { for (int j = -k; j <= k; ++j) { double value = static_cast<double>((1.0 / (2 * PI * sigma * sigma)) * (exp(-(i * i + j * j) / (2 * sigma * sigma)))); kernel[i + k][j + k] = value; sum += value; } } // 归一化 for (int i = 0; i < kSize; ++i) { for (int j = 0; j < kSize; ++j) { kernel[i][j] /= sum; } } return kernel; }

三、一维高斯滤波

同理,由高斯函数可知,我们可以分解为两个一维的滤波进行卷积计算,分离式滤波核正是opencv获得高斯滤波核的底层实现原理,这里我们也用底层代码实现,我们生成一维的卷积核,先用一维卷积核对图像行向量进行卷积,再用该滤波对图像的列向量进行卷积即可。代码如下:

// 分离式高斯滤波核 std::vector<double> GenerateSeparateGaussianKernel(int kSize, double sigma) { int k = kSize / 2; double sum = 0.0; std::vector<double>kernel(kSize, 0); const double PI = 3.14159265358979323846; for (int i = -k; i <= k; ++i) { double value = static_cast<double>((1.0 / (sqrt(2 * PI) * sigma)) * exp(-(i * i) / (sqrt(2) * sigma))); kernel[i + k] = value; sum += value; } // 归一化 for (int i = 0; i < kSize; ++i) { kernel[i] /= sum; } return kernel; }

四、滤波分离

这里我们重点讲解一下滤波分离,首先我们是知道高斯函数是可分离的。那对于一般的滤波核呢?我们不可以直接进行一维滤波计算,首先是否可分离,需要满足矩阵可分离性质,即满足矩阵 = 行 × 列,也就是秩为1。如果我们在计算位置滤波核的时候,不确定这个矩阵的秩是否为1,我们可以先用 SVD 计算一下它的秩,或者观察它的奇异值。如果除了第一个奇异值外,其他奇异值不为 0(或接近 0),那就不能分离。如果秩为1,我们可以采用以下方法对其进行分解,在核中找到任何一个非0元素,并将其值表示为E,形成行向量r和列向量c,他们分别等于E元素所在的那一行和那一列,这个时候这个r和c就是分离出来的行滤波核和列滤波核,原理如下。

代码如下:

// 将滤波核进行分离 void splitGaussianKernel(const std::vector<std::vector<double>>& GaussianKernel, std::vector<double>& c, std::vector<double>& r) { // 获取高斯核的行和列 int rows = GaussianKernel.size(); int cols = GaussianKernel[0].size(); r.resize(rows); c.resize(cols); // 找到非零元素,高斯核的中心就是最稳定的非零元素 double center = GaussianKernel[rows / 2][cols / 2]; // 提取非零元素所在行 for (int i = 0; i < rows; ++i) { r[i] = GaussianKernel[i][center]; } // 提取非零元素列 for (int j = 0; j < cols; ++j) { c[j] = GaussianKernel[center][j]; } }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/2/25 4:37:55

青少年运动员慢性踝关节不稳的四周踝关节康复计划

严正声明&#xff1a;本博客内容仅为学习使用&#xff0c;不具备任何医学建议或者参考价值。如有不适&#xff0c;请遵医嘱。本博客所转载之内容&#xff0c;不能作为正式的医学参考&#xff0c;仅供学习 青少年运动员慢性踝关节不稳的四周踝关节康复计划 Four-Week Ankle-Reh…

作者头像 李华
网站建设 2026/2/23 21:38:13

vue基于Springboot框架的新农村自建房改造管理系统

目录已开发项目效果实现截图开发技术系统开发工具&#xff1a;核心代码参考示例1.建立用户稀疏矩阵&#xff0c;用于用户相似度计算【相似度矩阵】2.计算目标用户与其他用户的相似度系统测试总结源码文档获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&…

作者头像 李华
网站建设 2026/2/26 0:13:03

基于C技术与SOCKET网络通信技术的局域网聊天系统

**# 基于C技术与SOCKET网络通信技术的局域网聊天系统 第一章 系统概述 在企业办公、校园协作等局域网场景中&#xff0c;传统即时通信工具依赖公网服务器&#xff0c;存在数据隐私泄露风险与网络延迟问题&#xff0c;而基于C技术与Socket网络通信的局域网聊天系统&#xff0c;通…

作者头像 李华
网站建设 2026/2/21 23:11:43

LobeChat实时流式输出实现原理剖析

LobeChat 实时流式输出实现原理剖析 在构建现代 AI 聊天应用的今天&#xff0c;用户早已不再满足于“发送问题、等待答案”的传统交互模式。当大语言模型&#xff08;LLM&#xff09;开始进入千家万户&#xff0c;用户体验的边界也被不断拉高——人们期望看到文字像人类打字一…

作者头像 李华
网站建设 2026/2/17 20:17:54

人人都在谈大模型,但90%的企业AI转型,都死在了数据这一关

从CEO到一线员工&#xff0c;几乎所有人都在热烈地讨论着大模型的最新进展和各种眼花缭乱的AI应用。我们仿佛进入了一个模型为王的时代&#xff0c;似乎只要接入最强的模型&#xff0c;就能解决所有问题。但现实是残酷的。 为什么很多企业AI项目总是做不出来&#xff1f; 我们也…

作者头像 李华
网站建设 2026/2/19 10:00:38

机器学习--线性回归

1、线性回归定义线性回归是利用数理统计中回归分析&#xff0c;来确定两种或两种以上变量间相互依赖的定量关系的一种统计分析方法。相关关系:包含因果关系和平行关系因果关系:回归分析【原因引起结果&#xff0c;需要明确自变量和因变量平行关系:相关分析【无因果关系&#xf…

作者头像 李华