0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

高斯濾波器的原理及其實現(xiàn)過程

新機器視覺 ? 來源:新機器視覺 ? 作者:新機器視覺 ? 2021-03-20 10:41 ? 次閱讀

本文主要介紹了高斯濾波器的原理及其實現(xiàn)過程 高斯濾波器是一種線性濾波器,能夠有效的抑制噪聲,平滑圖像。其作用原理和均值濾波器類似,都是取濾波器窗口內(nèi)的像素的均值作為輸出。其窗口模板的系數(shù)和均值濾波器不同,均值濾波器的模板系數(shù)都是相同的為1;而高斯濾波器的模板系數(shù),則隨著距離模板中心的增大而系數(shù)減小。所以,高斯濾波器相比于均值濾波器對圖像個模糊程度較小。 什么是高斯濾波器 既然名稱為高斯濾波器,那么其和高斯分布(正態(tài)分布)是有一定的關系的。一個二維的高斯函數(shù)如下:

cddf0d76-88ef-11eb-8b86-12bb97331649.png

其中(x,y)(x,y)為點坐標,在圖像處理中可認為是整數(shù);σσ是標準差。要想得到一個高斯濾波器的模板,可以對高斯函數(shù)進行離散化,得到的高斯函數(shù)值作為模板的系數(shù)。例如:要產(chǎn)生一個3×33×3的高斯濾波器模板,以模板的中心位置為坐標原點進行取樣。模板在各個位置的坐標,如下所示(x軸水平向右,y軸豎直向下)

這樣,將各個位置的坐標帶入到高斯函數(shù)中,得到的值就是模板的系數(shù)。 對于窗口模板的大小為(2k+1)×(2k+1),模板中各個元素值的計算公式如下:

ce5ab9ee-88ef-11eb-8b86-12bb97331649.png

這樣計算出來的模板有兩種形式:小數(shù)和整數(shù)。

小數(shù)形式的模板,就是直接計算得到的值,沒有經(jīng)過任何的處理;

整數(shù)形式的,則需要進行歸一化處理,將模板左上角的值歸一化為1,下面會具體介紹。使用整數(shù)的模板時,需要在模板的前面加一個系數(shù),系數(shù)為也就是模板系數(shù)和的倒數(shù)。

高斯模板的生成 知道模板生成的原理,實現(xiàn)起來也就不困難了

void generateGaussianTemplate(double window[][11], int ksize, double sigma){ static const double pi = 3.1415926; int center = ksize / 2; // 模板的中心位置,也就是坐標的原點 double x2, y2; for (int i = 0; i < ksize; i++) { x2 = pow(i - center, 2); for (int j = 0; j < ksize; j++) { y2 = pow(j - center, 2); double g = exp(-(x2 + y2) / (2 * sigma * sigma)); g /= 2 * pi * sigma; window[i][j] = g; } } double k = 1 / window[0][0]; // 將左上角的系數(shù)歸一化為1 for (int i = 0; i < ksize; i++) { for (int j = 0; j < ksize; j++) { window[i][j] *= k; } }} ? 需要一個二維數(shù)組,存放生成的系數(shù)(這里假設模板的最大尺寸不會超過11);第二個參數(shù)是模板的大小(不要超過11);第三個參數(shù)就比較重要了,是高斯分布的標準差。 生成的過程,首先根據(jù)模板的大小,找到模板的中心位置ksize/2。然后就是遍歷,根據(jù)高斯分布的函數(shù),計算模板中每個系數(shù)的值。 ? 需要注意的是,最后歸一化的過程,使用模板左上角的系數(shù)的倒數(shù)作為歸一化的系數(shù)(左上角的系數(shù)值被歸一化為1),模板中的每個系數(shù)都乘以該值(左上角系數(shù)的倒數(shù)),然后將得到的值取整,就得到了整數(shù)型的高斯濾波器模板。 ? 下面截圖生成的是,大小為3×3,σ=0.83×3,σ=0.8的模板 ?

對上述解結果取整后得到如下模板: ced64668-88ef-11eb-8b86-12bb97331649.png ? 這個模板就比較熟悉了,其就是根據(jù)σ=0.8的高斯函數(shù)生成的模板。 ? 至于小數(shù)形式的生成也比較簡單,去掉歸一化的過程,并且在求解過程后,模板的每個系數(shù)要除以所有系數(shù)的和。具體代碼如下:

void generateGaussianTemplate(double window[][11], int ksize, double sigma){ static const double pi = 3.1415926; int center = ksize / 2; // 模板的中心位置,也就是坐標的原點 double x2, y2; double sum = 0; for (int i = 0; i < ksize; i++) { x2 = pow(i - center, 2); for (int j = 0; j < ksize; j++) { y2 = pow(j - center, 2); double g = exp(-(x2 + y2) / (2 * sigma * sigma)); g /= 2 * pi * sigma; sum += g; window[i][j] = g; } } //double k = 1 / window[0][0]; // 將左上角的系數(shù)歸一化為1 for (int i = 0; i < ksize; i++) { for (int j = 0; j < ksize; j++) { window[i][j] /= sum; } }} ? 3×3,σ=0.8的小數(shù)型模板。

σσ值的意義及選取 通過上述的實現(xiàn)過程,不難發(fā)現(xiàn),高斯濾波器模板的生成最重要的參數(shù)就是高斯分布的標準差σσ。標準差代表著數(shù)據(jù)的離散程度,如果σσ較小,那么生成的模板的中心系數(shù)較大,而周圍的系數(shù)較小,這樣對圖像的平滑效果就不是很明顯;反之,σσ較大,則生成的模板的各個系數(shù)相差就不是很大,比較類似均值模板,對圖像的平滑效果比較明顯。 來看下一維高斯分布的概率分布密度圖:

橫軸表示可能得取值x,豎軸表示概率分布密度F(x),那么不難理解這樣一個曲線與x軸圍成的圖形面積為1。σσ(標準差)決定了這個圖形的寬度,可以得出這樣的結論:σσ越大,則圖形越寬,尖峰越小,圖形較為平緩;σσ越小,則圖形越窄,越集中,中間部分也就越尖,圖形變化比較劇烈。這其實很好理解,如果sigma也就是標準差越大,則表示該密度分布一定比較分散,由于面積為1,于是尖峰部分減小,寬度越寬(分布越分散);同理,當σσ越小時,說明密度分布較為集中,于是尖峰越尖,寬度越窄! 于是可以得到如下結論: σσ越大,分布越分散,各部分比重差別不大,于是生成的模板各元素值差別不大,類似于平均模板; σσ越小,分布越集中,中間部分所占比重遠遠高于其他部分,反映到高斯模板上就是中心元素值遠遠大于其他元素值,于是自然而然就相當于中間值得點運算。 基于OpenCV的實現(xiàn) 在生成高斯模板好,其簡單的實現(xiàn)和其他的空間濾波器沒有區(qū)別,具體代碼如下:

void GaussianFilter(const Mat &src, Mat &dst, int ksize, double sigma){ CV_Assert(src.channels() || src.channels() == 3); // 只處理單通道或者三通道圖像 const static double pi = 3.1415926; // 根據(jù)窗口大小和sigma生成高斯濾波器模板 // 申請一個二維數(shù)組,存放生成的高斯模板矩陣 double **templateMatrix = new double*[ksize]; for (int i = 0; i < ksize; i++) templateMatrix[i] = new double[ksize]; int origin = ksize / 2; // 以模板的中心為原點 double x2, y2; double sum = 0; for (int i = 0; i < ksize; i++) { x2 = pow(i - origin, 2); for (int j = 0; j < ksize; j++) { y2 = pow(j - origin, 2); // 高斯函數(shù)前的常數(shù)可以不用計算,會在歸一化的過程中給消去 double g = exp(-(x2 + y2) / (2 * sigma * sigma)); sum += g; templateMatrix[i][j] = g; } } for (int i = 0; i < ksize; i++) { for (int j = 0; j < ksize; j++) { templateMatrix[i][j] /= sum; cout << templateMatrix[i][j] << " "; } cout << endl; } // 將模板應用到圖像中 int border = ksize / 2; copyMakeBorder(src, dst, border, border, border, border, BorderTypes::BORDER_REFLECT); int channels = dst.channels(); int rows = dst.rows - border; int cols = dst.cols - border; for (int i = border; i < rows; i++) { for (int j = border; j < cols; j++) { double sum[3] = { 0 }; for (int a = -border; a <= border; a++) { for (int b = -border; b <= border; b++) { if (channels == 1) { sum[0] += templateMatrix[border + a][border + b] * dst.at(i + a, j + b); } else if (channels == 3) { Vec3b rgb = dst.at(i + a, j + b); auto k = templateMatrix[border + a][border + b]; sum[0] += k * rgb[0]; sum[1] += k * rgb[1]; sum[2] += k * rgb[2]; } } } for (int k = 0; k < channels; k++) { if (sum[k] < 0) sum[k] = 0; else if (sum[k] > 255) sum[k] = 255; } if (channels == 1) dst.at(i, j) = static_cast(sum[0]); else if (channels == 3) { Vec3b rgb = { static_cast(sum[0]), static_cast(sum[1]), static_cast(sum[2]) }; dst.at(i, j) = rgb; } } } // 釋放模板數(shù)組 for (int i = 0; i < ksize; i++) delete[] templateMatrix[i]; delete[] templateMatrix;} ? 只處理單通道或者三通道圖像,模板生成后,其濾波(卷積過程)就比較簡單了。不過,這樣的高斯濾波過程,其循環(huán)運算次數(shù)為m×n×ksize2,其中m,n為圖像的尺寸;ksize為高斯濾波器的尺寸。這樣其時間復雜度為O(ksize2),隨濾波器的模板的尺寸呈平方增長,當高斯濾波器的尺寸較大時,其運算效率是極低的。為了,提高濾波的運算速度,可以將二維的高斯濾波過程分解開來。 ?分離實現(xiàn)高斯濾波? 由于高斯函數(shù)的可分離性,尺寸較大的高斯濾波器可以分成兩步進行:首先將圖像在水平(豎直)方向與一維高斯函數(shù)進行卷積;然后將卷積后的結果在豎直(水平)方向使用相同的一維高斯函數(shù)得到的模板進行卷積運算。具體實現(xiàn)代碼如下:

// 分離的計算void separateGaussianFilter(const Mat &src, Mat &dst, int ksize, double sigma){ CV_Assert(src.channels()==1 || src.channels() == 3); // 只處理單通道或者三通道圖像 // 生成一維的高斯濾波模板 double *matrix = new double[ksize]; double sum = 0; int origin = ksize / 2; for (int i = 0; i < ksize; i++) { // 高斯函數(shù)前的常數(shù)可以不用計算,會在歸一化的過程中給消去 double g = exp(-(i - origin) * (i - origin) / (2 * sigma * sigma)); sum += g; matrix[i] = g; } // 歸一化 for (int i = 0; i < ksize; i++) matrix[i] /= sum; // 將模板應用到圖像中 int border = ksize / 2; copyMakeBorder(src, dst, border, border, border, border, BorderTypes::BORDER_REFLECT); int channels = dst.channels(); int rows = dst.rows - border; int cols = dst.cols - border; // 水平方向 for (int i = border; i < rows; i++) { for (int j = border; j < cols; j++) { double sum[3] = { 0 }; for (int k = -border; k <= border; k++) { if (channels == 1) { sum[0] += matrix[border + k] * dst.at(i, j + k); // 行不變,列變化;先做水平方向的卷積 } else if (channels == 3) { Vec3b rgb = dst.at(i, j + k); sum[0] += matrix[border + k] * rgb[0]; sum[1] += matrix[border + k] * rgb[1]; sum[2] += matrix[border + k] * rgb[2]; } } for (int k = 0; k < channels; k++) { if (sum[k] < 0) sum[k] = 0; else if (sum[k] > 255) sum[k] = 255; } if (channels == 1) dst.at(i, j) = static_cast(sum[0]); else if (channels == 3) { Vec3b rgb = { static_cast(sum[0]), static_cast(sum[1]), static_cast(sum[2]) }; dst.at(i, j) = rgb; } } } // 豎直方向 for (int i = border; i < rows; i++) { for (int j = border; j < cols; j++) { double sum[3] = { 0 }; for (int k = -border; k <= border; k++) { if (channels == 1) { sum[0] += matrix[border + k] * dst.at(i + k, j); // 列不變,行變化;豎直方向的卷積 } else if (channels == 3) { Vec3b rgb = dst.at(i + k, j); sum[0] += matrix[border + k] * rgb[0]; sum[1] += matrix[border + k] * rgb[1]; sum[2] += matrix[border + k] * rgb[2]; } } for (int k = 0; k < channels; k++) { if (sum[k] < 0) sum[k] = 0; else if (sum[k] > 255) sum[k] = 255; } if (channels == 1) dst.at(i, j) = static_cast(sum[0]); else if (channels == 3) { Vec3b rgb = { static_cast(sum[0]), static_cast(sum[1]), static_cast(sum[2]) }; dst.at(i, j) = rgb; } } } delete[] matrix;} 代碼沒有重構較長,不過其實現(xiàn)原理是比較簡單的。首先得到一維高斯函數(shù)的模板,在卷積(濾波)的過程中,保持行不變,列變化,在水平方向上做卷積運算;接著在上述得到的結果上,保持列不邊,行變化,在豎直方向上做卷積運算。這樣分解開來,算法的時間復雜度為O(ksize)O(ksize),運算量和濾波器的模板尺寸呈線性增長。 在OpenCV也有對高斯濾波器的封裝GaussianBlur,其聲明如下:

CV_EXPORTS_W void GaussianBlur( InputArray src, OutputArray dst, Size ksize, double sigmaX, double sigmaY = 0, int borderType = BORDER_DEFAULT ); 二維高斯函數(shù)的標準差在x和y方向上應該分別有一個標準差,在上面的代碼中一直設其在x和y方向的標準是相等的,在OpenCV中的高斯濾波器中,可以在x和y方向上設置不同的標準差。 下圖是自己實現(xiàn)的高斯濾波器和OpenCV中的GaussianBlur的結果對比

上圖是5×5,σ=0.8的高斯濾波器,可以看出兩個實現(xiàn)得到的結果沒有很大的區(qū)別。 總結 高斯濾波器是一種線性平滑濾波器,其濾波器的模板是對二維高斯函數(shù)離散得到。由于高斯模板的中心值最大,四周逐漸減小,其濾波后的結果相對于均值濾波器來說更好。 高斯濾波器最重要的參數(shù)就是高斯分布的標準差σσ,標準差和高斯濾波器的平滑能力有很大的能力,σσ越大,高斯濾波器的頻帶就較寬,對圖像的平滑程度就越好。通過調(diào)節(jié)σσ參數(shù),可以平衡對圖像的噪聲的抑制和對圖像的模糊。

責任編輯:lq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 濾波器
    +關注

    關注

    161

    文章

    7854

    瀏覽量

    178520
  • 模板
    +關注

    關注

    0

    文章

    108

    瀏覽量

    20582
  • 函數(shù)
    +關注

    關注

    3

    文章

    4344

    瀏覽量

    62810

原文標題:高斯濾波器的原理及其實現(xiàn)過程(附模板代碼)

文章出處:【微信號:vision263com,微信公眾號:新機器視覺】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    圖像高斯濾波的原理及FPGA實現(xiàn)思路

    1.概念 高斯分布 圖像濾波高斯濾波介紹 圖像處理算法|高斯濾波
    的頭像 發(fā)表于 12-07 09:12 ?412次閱讀
    圖像<b class='flag-5'>高斯</b><b class='flag-5'>濾波</b>的原理及FPGA<b class='flag-5'>實現(xiàn)</b>思路

    高斯濾波和均值濾波的區(qū)別

    高斯濾波和均值濾波在圖像處理中都是常用的平滑濾波方法,但它們之間存在一些關鍵的區(qū)別。以下是兩者之間的主要區(qū)別: 1. 濾波原理
    的頭像 發(fā)表于 09-29 09:40 ?777次閱讀

    高斯濾波和雙邊濾波的區(qū)別

    高斯濾波和雙邊濾波在圖像處理中都是常用的平滑濾波技術,但它們之間存在一些顯著的區(qū)別。以下是兩者之間的主要區(qū)別: 一、基本原理 高斯
    的頭像 發(fā)表于 09-29 09:37 ?532次閱讀

    高斯濾波的卷積核怎么確定

    N為一個奇數(shù),如3、5、7等。奇數(shù)大小的卷積核有助于確定一個中心像素點,便于計算。 大小選擇 :卷積核的大小決定了濾波器的范圍。較大的卷積核可以覆蓋更多的像素點,從而更好地平滑圖像,但也可能導致圖像細節(jié)丟失過多。因此,在
    的頭像 發(fā)表于 09-29 09:29 ?679次閱讀

    高斯濾波的基本原理有哪些

    高斯濾波的基本原理可以從以下幾個方面進行闡述: 一、定義與性質(zhì) 定義 :高斯濾波(Gaussian Filter)是一種常見的圖像處理技術,實質(zhì)上是一種信號的
    的頭像 發(fā)表于 09-29 09:27 ?675次閱讀

    emi濾波器是什么濾波器

    會對電子設備的正常工作產(chǎn)生影響,甚至導致設備損壞。因此,EMI濾波器在電子設備中具有非常重要的作用。 EMI濾波器的原理 EMI濾波器的基本原理是利用濾波器對電磁波進行選擇性傳輸或阻斷
    的頭像 發(fā)表于 08-25 15:59 ?1154次閱讀

    陷波濾波器原理及其作用是什么

    是通過特定的電路設計,實現(xiàn)對特定頻率信號的抑制或消除。其核心思想是利用電路的諧振特性,使得特定頻率的信號在電路中產(chǎn)生相位差,從而實現(xiàn)信號的抵消。 1.1 諧振原理 陷波濾波器的工作原理基于諧振原理。當電路中的電感和
    的頭像 發(fā)表于 08-21 14:14 ?2597次閱讀

    iir濾波器和fir濾波器的優(yōu)勢和特點

    (Infinite Impulse Response)濾波器是一種具有無限脈沖響應的數(shù)字濾波器。它的輸出不僅取決于當前的輸入值,還取決于之前的輸入值。IIR濾波器通常由遞歸濾波器
    的頭像 發(fā)表于 07-19 09:28 ?1737次閱讀

    濾波器原理及其作用 濾波器電路圖分析

    信號中的不同頻率成分。這可以通過多種方式實現(xiàn),包括模擬濾波器和數(shù)字濾波器濾波器的原理 頻率響應概念 濾波器對不同頻率信號的響應 頻率選擇
    的頭像 發(fā)表于 06-20 15:59 ?9664次閱讀
    <b class='flag-5'>濾波器</b>原理<b class='flag-5'>及其</b>作用 <b class='flag-5'>濾波器</b>電路圖分析

    什么是低通濾波器?低通濾波器有什么作用?

    在電子工程領域中,濾波器是一種用于信號處理的重要元件。而低通濾波器作為濾波器的一種類型,具有其獨特的頻率響應特性。本文維愛普電源濾波器小編將詳細探討低通
    的頭像 發(fā)表于 04-08 16:30 ?3959次閱讀
    什么是低通<b class='flag-5'>濾波器</b>?低通<b class='flag-5'>濾波器</b>有什么作用?

    高通濾波器、低通濾波器、帶通濾波器怎樣測幅頻特性?

    高通濾波器、低通濾波器、帶通濾波器怎樣測幅頻特性? 高通濾波器、低通濾波器和帶通濾波器是常用的
    的頭像 發(fā)表于 03-28 17:28 ?5411次閱讀

    濾波器的選用、安裝及其

    濾波器,想必做電子的同行對這個器件再熟悉不過了,這是一個非常常見的,而且使用相當頻繁的元件。 特別是涉及射頻領域更是如此,可以毫不夸張的說,做射頻電路就是學會使用各種濾波器的電路。因為芯片
    的頭像 發(fā)表于 03-06 10:19 ?915次閱讀
    <b class='flag-5'>濾波器</b>的選用、安裝<b class='flag-5'>及其</b>它

    濾波器分類及其特點 濾波器電路的作用

    濾波器是一種可以改變信號頻率特性的電路,它可以通過選擇性地通過或抑制不同頻率的信號來實現(xiàn)對特定頻率信號的處理。不同類型的濾波器根據(jù)其頻率響應和特點可以分為許多不同的類型,下面將詳細介紹常見的
    的頭像 發(fā)表于 02-03 10:22 ?1964次閱讀

    什么是電源濾波器?如何安裝電源濾波器?

    什么是電源濾波器?如何安裝電源濾波器? 電源濾波器是一種電子設備,通常被用于凈化電源中的噪聲和干擾。在電源中,噪聲和干擾是由各種原因產(chǎn)生的,包括電網(wǎng)擾動,設備之間的相互干擾以及其他外部
    的頭像 發(fā)表于 02-02 14:41 ?1065次閱讀

    Pi濾波器的Python實現(xiàn)示例

    Pi 濾波器是一種出色的低通濾波器,與傳統(tǒng)的 LC Pi 濾波器有很大不同。當 Pi 濾波器設計用于低通時,輸出保持穩(wěn)定且 k 系數(shù)恒定。
    的頭像 發(fā)表于 02-01 14:35 ?1085次閱讀
    Pi<b class='flag-5'>濾波器</b>的Python<b class='flag-5'>實現(xiàn)</b>示例