Bilateral Filtering for Gray and Color Images

双边滤波器:保留边界的平滑滤波器。 在局部上,就是在灰度值差异不大的区域平滑,在灰度值差异比较大的边界地区保留边界。所以双边滤波器作用于每个像素的同时,必然会受到领域像素点的距离、灰度值差的权重影响。

已知低通滤波可以表示为:

range filter可以表示为:(range filter 试选定一个数值范围,再做滤波的一个操作)

所以,双边滤波器的定义是:

其中,k(x)是归一化(normalize)函数,

( f 表示原图像,h 表示处理后的图像 x 表示 h 中某个像素点位置,ξ 表示 f 中x位置像素点的邻域像素,f(ξ)表示该像素点的灰度值,c表示低通滤波, s表示range filter)

其中,

//Filters.h

#ifndef FILTERS_H
#define FILTERS_H #include "opencv2/imgproc.hpp"
#include "opencv2/highgui.hpp"
#include "opencv2/core.hpp" #include <iostream>
#include <cmath> //Bilateral Filtering
//sigmaD == sigmaSpace, sigmaR == sigmaColor
cv::Mat BilateralFilter(cv::Mat inputImg, int filterSize, double sigmaD, double sigmaR); cv::Mat fastBilateralFilter(cv::Mat inputImg, int filterSize, double sigmaD, double sigmaR); #endif // ! FILTERS_H
//Filters.cpp

#include "Filters.h"

double SpaceFactor(int x1, int y1, int x2, int y2, double sigmaD) {
double absX = pow(abs(x1 - x2), );
double absY = pow(abs(y1 - y2), ); return exp(-(absX + absY) / ( * pow(sigmaD, )));
} double ColorFactor(int x, int y, double sigmaR) {
double distance = abs(x - y) / sigmaR;
return exp(-0.5 * pow(distance, ));
} cv::Mat BilateralFilter(cv::Mat inputImg, int filterSize, double sigmaD, double sigmaR) {
int len; //must be odd number
cv::Mat gray; // must be 1-channel image
cv::Mat LabImage; // if channels == 3 if (filterSize % != || filterSize <= ) {
std::cerr << "Filter Size must be a positive odd number!" << std::endl;
return inputImg;
}
len = filterSize / ; if (inputImg.channels() >= ) {
cv::cvtColor(inputImg, LabImage, cv::COLOR_BGR2Lab);
gray = cv::Mat::zeros(LabImage.size(), CV_8UC1);
for (int i = ; i < LabImage.rows; i++) {
for (int j = ; j < LabImage.cols; j++) {
gray.ptr<uchar>(i)[j] = LabImage.ptr<uchar>(i, j)[];
}
}
}
else if(inputImg.channels() == ){
inputImg.copyTo(gray);
}
else {
std::cerr << "the count of input image's channel can not be 2!" << std::endl;
return inputImg;
} cv::Mat resultGrayImg = cv::Mat::zeros(gray.size(), CV_8UC1);
for (int i = ; i < gray.rows; i++) {
for (int j = ; j < gray.cols; j++) {
double k = ;
double f = ;
for (int r = i - len; r <= i + len; r++) {
for (int c = j - len; c <= j + len; c++) {
if (r < || c < || r >= gray.rows || c >= gray.cols)
continue;
f = f + gray.ptr<uchar>(r)[c] * SpaceFactor(i, j, r, c, sigmaD) * ColorFactor(gray.ptr<uchar>(i)[j], gray.ptr<uchar>(r)[c], sigmaD);
k += SpaceFactor(i, j, r, c, sigmaD) * ColorFactor(gray.ptr<uchar>(i)[j], gray.ptr<uchar>(r)[c], sigmaD);
}
}
int value = f / k;
if (value < ) value = ;
else if (value > ) value = ; resultGrayImg.ptr<uchar>(i)[j] = (uchar)value;
}
} cv::Mat resultImg;
if (inputImg.channels() >= ) {
for (int i = ; i < LabImage.rows; i++) {
for (int j = ; j < LabImage.cols; j++) {
LabImage.ptr<uchar>(i, j)[] = resultGrayImg.ptr<uchar>(i)[j];
}
}
cv::cvtColor(LabImage, resultImg, cv::COLOR_Lab2BGR);
}
else {
resultGrayImg.copyTo(resultImg);
} return resultImg;
} cv::Mat fastBilateralFilter(cv::Mat inputImg, int filterSize, double sigmaD, double sigmaR) {
int len; //must be odd number
cv::Mat gray; // must be 1-channel image
cv::Mat LabImage; // if channels == 3 if (filterSize % != || filterSize <= ) {
std::cerr << "Filter Size must be a positive odd number!" << std::endl;
return inputImg;
}
len = filterSize / ; if (inputImg.channels() >= ) {
cv::cvtColor(inputImg, LabImage, cv::COLOR_BGR2Lab);
gray = cv::Mat::zeros(LabImage.size(), CV_8UC1);
for (int i = ; i < LabImage.rows; i++) {
for (int j = ; j < LabImage.cols; j++) {
gray.ptr<uchar>(i)[j] = LabImage.ptr<uchar>(i, j)[];
}
}
}
else if (inputImg.channels() == ) {
inputImg.copyTo(gray);
}
else {
std::cerr << "the count of input image's channel can not be 2!" << std::endl;
return inputImg;
} cv::Mat resultGrayImg = cv::Mat::zeros(gray.size(), CV_8UC1);
for (int i = ; i < gray.rows; i++) {
for (int j = ; j < gray.cols; j++) {
double k = ;
double f = ;
double sum = ;
for (int r = i - len; r <= i + len; r++) {
if (r < || r >= gray.rows)
continue;
f = f + gray.ptr<uchar>(r)[j] * SpaceFactor(i, j, r, j, sigmaD) * ColorFactor(gray.ptr<uchar>(i)[j], gray.ptr<uchar>(r)[j], sigmaD);
k += SpaceFactor(i, j, r, j, sigmaD) * ColorFactor(gray.ptr<uchar>(i)[j], gray.ptr<uchar>(r)[j], sigmaD);
}
sum = f / k;
f = k = 0.0;
for (int c = j - len; c <= j + len; c++) {
if (c < || c >= gray.cols)
continue;
f = f + gray.ptr<uchar>(i)[c] * SpaceFactor(i, j, i, c, sigmaD) * ColorFactor(gray.ptr<uchar>(i)[j], gray.ptr<uchar>(i)[c], sigmaD);
k += SpaceFactor(i, j, i, c, sigmaD) * ColorFactor(gray.ptr<uchar>(i)[j], gray.ptr<uchar>(i)[c], sigmaD);
}
int value = (sum + f / k) / ;
if (value < ) value = ;
else if (value > ) value = ; resultGrayImg.ptr<uchar>(i)[j] = (uchar)value;
}
} cv::Mat resultImg;
if (inputImg.channels() >= ) {
for (int i = ; i < LabImage.rows; i++) {
for (int j = ; j < LabImage.cols; j++) {
LabImage.ptr<uchar>(i, j)[] = resultGrayImg.ptr<uchar>(i)[j];
}
}
cv::cvtColor(LabImage, resultImg, cv::COLOR_Lab2BGR);
}
else {
resultGrayImg.copyTo(resultImg);
} return resultImg;
}
//main.cpp

#include <iostream>
#include <time.h> #include "Filters.h" using namespace std; int main() {
cv::Mat img = cv::imread("Capture.jpg", cv::IMREAD_UNCHANGED);
clock_t begin_time = clock();
cv::Mat result = BilateralFilter(img, , 12.5, );
std::cout << float(clock() - begin_time) / CLOCKS_PER_SEC << std:: endl;
cv::imshow("original", result);
cv::waitKey();
cv::imwrite("original.jpg", result); begin_time = clock();
result = fastBilateralFilter(img, , 12.5, );
std::cout << float(clock() - begin_time) / CLOCKS_PER_SEC << std::endl;
cv::imshow("fast", result);
cv::waitKey();
cv::imwrite("fast.jpg", result); begin_time = clock();
cv::bilateralFilter(img, result, , , 12.5);
std::cout << float(clock() - begin_time) / CLOCKS_PER_SEC << std::endl;
cv::imshow("opencv", result);
cv::waitKey();
cv::imwrite("opencv.jpg", result); system("pause");
return ;
}

运行结果:

46.889s  5.694s  0.202s

二维算子降成两个一维算子之后,速度加快了一些,但是还是不如opencv的快,效果也比它差一些(No more reinventing the wheel...)

从左至右:“小雀斑”帅气原图、BilateralFilter处理结果、fastBilateralFilter处理结果、opencv接口处理结果

   

【C++】双边滤波器(bilateral filter)的更多相关文章

  1. 【VS开发】【图像处理】双边滤波器bilateral filter

    目录(?)[-] 简介 原理 代码实现 1 Spatial Weight 2 Similarity Weight 3 Color Filtering 在SSAO中的使用 1. 简介 图像平滑是一个重要 ...

  2. Bilateral Filter

    最近在看图像风格化的论文的时候,频繁遇到 Bilateral Filter.google 一波后,发现并不是什么不得了的东西,但它的思想却很有借鉴意义. 简介 Bilateral Filter,中文又 ...

  3. 三维网格去噪算法(bilateral filter)

    受图像双边滤波算法的启发,[Fleishman et al. 2003]和[Jones et al. 2003]分别提出了利用双边滤波算法对噪声网格进行光顺去噪的算法,两篇文章都被收录于当年的SIGG ...

  4. vs2015+opencv3.3.1 实现 c++ 双边滤波器(Bilateral Filter)

    #include <opencv2\highgui\highgui.hpp> #include <iostream> #include<vector> using ...

  5. 卡尔曼滤波器 Kalman Filter (转载)

    在学习卡尔曼滤波器之前,首先看看为什么叫“卡尔曼”.跟其他著名的理论(例如傅立叶变换,泰勒级数等等)一样,卡尔曼也是一个人的名字,而跟他们不同的是,他是个现代人! 卡 尔曼全名Rudolf Emil ...

  6. [转]计算机视觉之跟踪算法——相关滤波器Correlation Filter

    https://blog.csdn.net/victoriaw/article/details/62416759 ASEF相关滤波器: Average of Synthetic Exact Filte ...

  7. 灰度图像--图像增强 双边滤波 Bilateral Filtering

    学习DIP第31天 转载请标明本文出处:http://blog.csdn.net/tonyshengtan,欢迎大家转载,发现博客被某些论坛转载后,图像无法正常显示,无法正常表达本人观点,对此表示很不 ...

  8. 测试卡尔曼滤波器(Kalman Filter)

    真实的温度测试数据,通过加热棒加热一盆水测得的真实数据,X轴是时间秒,Y轴是温度: 1)滤波前 2)滤波后(p=10, q=0.0001, r=0.05, kGain=0;) 2)滤波后(p=10, ...

  9. Atitit   图像处理 平滑 也称 模糊, 归一化块滤波、高斯滤波、中值滤波、双边滤波)

    Atitit   图像处理 平滑 也称 模糊, 归一化块滤波.高斯滤波.中值滤波.双边滤波) 是一项简单且使用频率很高的图像处理方法 用途 去噪 去雾 各种线性滤波器对图像进行平滑处理,相关OpenC ...

随机推荐

  1. 「NOIP2005」「Codevs1106」篝火晚会

    题目描述 Description 佳佳刚进高中,在军训的时候,由于佳佳吃苦耐劳,很快得到了教官的赏识,成为了“小教官”.在军训结束的那天晚上,佳佳被命令组织同学们进行篝火晚会.一共有n个同学,编号从1 ...

  2. 【旧文章搬运】Windbg+Vmware驱动调试入门(一)---Windbg的设置

    原文发表于百度空间,2009-01-08========================================================================== Windb ...

  3. #ifndef <标识> #define <标识> #endif

    头件的中的#ifndef,这是一个很关键的东西.比如你有两个C文件,这两个C文件都include了同一个头文件.而编译时,这两个C文件要一同编译成一个可运行文件,于是问题来了,大量的声明冲突. 还是把 ...

  4. UVaLive 6950 && Gym 100299K Digraphs (DFS找环或者是找最长链)

    题意:有n个只包含两个字母的字符串, 要求构造一个m*m的字母矩阵, 使得矩阵的每行每列都不包含所给的字符串, m要尽量大, 如果大于20的话构造20*20的矩阵就行了. 析:开始吧,并没有读对题意, ...

  5. 洛谷 - P2181 - 对角线 - 打表 - 组合数学

    https://www.luogu.org/problemnew/show/P2181 对于某条对角线,除去从两端出发的对角线,其他的都与它有1个交点. 每个点有(n-3)条对角线,每条对角线和其余C ...

  6. poj 3648 Wedding【2-SAT+tarjan+拓扑】

    看错题*n,注意是输出新娘这边的-- 按2-SAT规则连互斥的边,然后注意连一条(1,1+n)表示新娘必选 然后输出color[belong[i]]==color[belong[1+n(新娘)]]的点 ...

  7. bzoj 1082: [SCOI2005]栅栏【二分+dfs】

    二分答案,dfs判断是否可行,当b[k]==b[k-1]时可以剪枝也就是后移枚举位置 #include<iostream> #include<cstdio> #include& ...

  8. 树的直径初探+Luogu P3629 [APIO2010]巡逻【树的直径】By cellur925

    题目传送门 我们先来介绍一个概念:树的直径. 树的直径:树中最远的两个节点间的距离.(树的最长链)树的直径有两种方法,都是$O(N)$. 第一种:两遍bfs/dfs(这里写的是两遍bfs) 从任意一个 ...

  9. Codeforces 1107E(区间dp)

    用solve(l, r, prefix)代表区间l开始r结束.带了prefix个前缀str[l](即l前面的串化简完压缩成prefix-1个str[l],加上str[l]共有prefix个)的最大值. ...

  10. 逆序数 HDOJ 4911 Inversion

    题目传送门 题意:可以交换两个相邻的数字顺序k次,问最后逆序对最少有多少 分析:根据逆序数的定理如果逆序数大于0,那么必定存在1<=i<n使得i和i+1交换后逆序数减1假设原逆序数为cnt ...