c/c++中opencv雙邊濾波的實現(xiàn)
雙邊濾波器(Bilateral Filter)是一種非線性的圖像平滑技術(shù),它能夠在有效去除噪聲的同時,較好地保留圖像的邊緣信息。這使其在許多圖像處理任務(wù)中非常受歡迎,例如圖像美化、降噪等。本文將介紹雙邊濾波的原理,并展示如何使用 C++ 和 OpenCV 庫來實現(xiàn)它。
雙邊濾波原理簡介
傳統(tǒng)的線性平滑濾波器(如高斯模糊、均值模糊)在平滑圖像時,不區(qū)分像素間的差異,導(dǎo)致邊緣區(qū)域也會被模糊掉。雙邊濾波器則巧妙地結(jié)合了空間鄰近度和像素值相似度(顏色強(qiáng)度)兩個方面的信息。
它主要由兩個高斯函數(shù)組成:
- 空間域高斯核 (Spatial Gaussian):與標(biāo)準(zhǔn)高斯模糊類似,根據(jù)像素間的空間距離來分配權(quán)重。距離中心像素越近的點,權(quán)重越大。
- 值域/顏色域高斯核 (Range/Intensity Gaussian):根據(jù)像素值(灰度值或顏色值)的差異來分配權(quán)重。與中心像素顏色越相似的鄰近像素,權(quán)重越大。
最終,一個像素的權(quán)重是這兩個高斯權(quán)重的乘積。這意味著,只有當(dāng)一個鄰近像素同時滿足“離得近”和“顏色像”這兩個條件時,它才會對中心像素的濾波結(jié)果產(chǎn)生較大影響。因此,邊緣(顏色差異大的地方)就能被較好地保留下來。
OpenCV中的 cv::bilateralFilter 函數(shù)
OpenCV 提供了 cv::bilateralFilter
函數(shù)來實現(xiàn)雙邊濾波。
函數(shù)原型 (C++):
void cv::bilateralFilter( cv::InputArray src, // 輸入圖像 (8位或浮點型, 1通道或3通道) cv::OutputArray dst, // 輸出圖像,與輸入圖像大小和類型相同 int d, // 濾波過程中每個像素鄰域的直徑。如果是非正數(shù),則由 sigmaSpace 計算得出 double sigmaColor, // 顏色空間濾波器的 sigma 值。這個參數(shù)越大,表明該像素鄰域內(nèi)有更寬廣的顏色會被混合到一起,產(chǎn)生較大的半相等顏色區(qū)域。 double sigmaSpace, // 坐標(biāo)空間濾波器的 sigma 值。這個參數(shù)越大,表明越遠(yuǎn)的像素會相互影響,從而使更大的區(qū)域中足夠相似的顏色獲取相同的顏色。當(dāng) d > 0 時,d 指定了鄰域大小且與 sigmaSpace 無關(guān)。否則,d 正比于 sigmaSpace。 int borderType = cv::BORDER_DEFAULT // 用于推斷圖像外部像素的某種邊界模式 );
參數(shù)詳解:
src
: 輸入圖像,可以是單通道灰度圖或三通道彩色圖。dst
: 輸出圖像,其大小和類型與輸入圖像src
相同。d
: 在濾波期間使用的每個像素鄰域的直徑。- 如果
d
為非正數(shù) (例如 0 或 -1),則會從sigmaSpace
計算得到一個合適的值。 - 通常建議將
d
設(shè)置為 5 到 9 之間的一個奇數(shù)。對于離線處理中較大的sigmaSpace
,可以考慮更大的d
值。
- 如果
sigmaColor
: 顏色標(biāo)準(zhǔn)差。該值越大,意味著濾波器會將更寬范圍內(nèi)的顏色認(rèn)為是相似的,從而導(dǎo)致更大區(qū)域的顏色趨于一致。較小的值則保留更多顏色細(xì)節(jié)。sigmaSpace
: 空間標(biāo)準(zhǔn)差。該值越大,意味著在計算當(dāng)前像素值時,會考慮更遠(yuǎn)距離的像素。如果d > 0
,則此參數(shù)對鄰域大小沒有影響;否則,鄰域大小與sigmaSpace
成正比。borderType
: 邊界處理方式,默認(rèn)為cv::BORDER_DEFAULT
。
C++/OpenCV 代碼示例
下面是一個簡單的 C++ 示例,演示如何加載一張圖片,對其應(yīng)用雙邊濾波,并顯示原圖和處理后的圖像。
#include <opencv2/opencv.hpp> // 包含OpenCV主要頭文件 #include <iostream> // 用于標(biāo)準(zhǔn)輸入輸出 int main() { // --- 1. 加載圖像 --- std::string imagePath = "your_image.jpg"; // 替換為你的圖片路徑 cv::Mat srcImage = cv::imread(imagePath); // 檢查圖像是否成功加載 if (srcImage.empty()) { std::cerr << "Error: Could not open or find the image!" << std::endl; return -1; } // --- 2. 應(yīng)用雙邊濾波 --- cv::Mat filteredImage; int diameter = 9; // 鄰域直徑 double sigmaColor = 75; // 顏色標(biāo)準(zhǔn)差 double sigmaSpace = 75; // 空間標(biāo)準(zhǔn)差 // 調(diào)用雙邊濾波函數(shù) cv::bilateralFilter(srcImage, filteredImage, diameter, sigmaColor, sigmaSpace); // --- 3. 顯示圖像 --- // 創(chuàng)建窗口 cv::namedWindow("Original Image", cv::WINDOW_AUTOSIZE); cv::namedWindow("Bilateral Filtered Image", cv::WINDOW_AUTOSIZE); // 顯示圖像 cv::imshow("Original Image", srcImage); cv::imshow("Bilateral Filtered Image", filteredImage); // --- 4. 等待按鍵并關(guān)閉窗口 --- cv::waitKey(0); // 等待用戶按下任意鍵 cv::destroyAllWindows(); //銷毀所有創(chuàng)建的窗口 return 0; }
編譯和運行:
確保你已經(jīng)正確安裝了 OpenCV。你可以使用 CMake 或者直接用 g++ 編譯 (鏈接 OpenCV 庫):
g++ your_code.cpp -o bilateral_filter_app $(pkg-config --cflags --libs opencv4) ./bilateral_filter_app
(將 your_code.cpp
替換為你的源文件名,opencv4
可能需要根據(jù)你的 OpenCV 版本調(diào)整為 opencv
或其他)
參數(shù)調(diào)節(jié)技巧 ??
正確選擇 d
, sigmaColor
, 和 sigmaSpace
的值對于獲得理想的濾波效果至關(guān)重要:
d
(鄰域直徑):- 通常設(shè)為奇數(shù),如 5, 7, 9。
- 如果設(shè)置為 0 或負(fù)數(shù),OpenCV 會根據(jù)
sigmaSpace
自動計算。 d
越大,計算量越大,濾波越慢。對于實時應(yīng)用,較小的d
(如 5) 通常是首選。
sigmaColor
(顏色標(biāo)準(zhǔn)差):- 此值控制顏色相似性的容忍度。
- 值越大:更多顏色會被認(rèn)為是相似的,圖像會被平滑得更厲害,產(chǎn)生類似卡通畫的效果,但可能丟失細(xì)微的顏色變化。
- 值越小:只有顏色非常接近的像素才會被一起平滑,邊緣保留更好,但降噪效果可能減弱。
- 建議范圍:10 ~ 150,根據(jù)具體圖像調(diào)整。
sigmaSpace
(空間標(biāo)準(zhǔn)差):- 此值控制空間鄰近性的影響范圍。
- 值越大:更遠(yuǎn)處的像素也會對當(dāng)前像素產(chǎn)生影響(前提是它們的顏色也相似),這有助于平滑較大的同色區(qū)域。
- 值越小:只有非常近的像素才會互相影響。
- 如果
d
已經(jīng)設(shè)置了一個正值,sigmaSpace
的影響主要體現(xiàn)在權(quán)重計算上,而不是鄰域大小。 - 建議范圍:10 ~ 150,根據(jù)具體圖像調(diào)整。
一般建議:
- 從小值開始嘗試,逐步增大,觀察效果。
sigmaColor
和sigmaSpace
通??梢栽O(shè)置為相近的值作為起點。- 如果圖像噪聲較大,可以適當(dāng)增大
sigmaColor
。 - 如果希望保留非常精細(xì)的紋理,應(yīng)使用較小的
sigmaColor
和sigmaSpace
。
優(yōu)缺點總結(jié)
優(yōu)點:
- ?? 邊緣保留:雙邊濾波最顯著的優(yōu)點是在去除噪聲的同時能很好地保留圖像的邊緣信息。
- ?? 表面平滑:對于圖像中的平坦區(qū)域,它可以有效地平滑噪聲。
缺點:
- ?? 計算量大:相比于高斯模糊等線性濾波器,雙邊濾波的計算復(fù)雜度更高,處理速度較慢。
- ?? 參數(shù)敏感:濾波效果對參數(shù)
d
,sigmaColor
,sigmaSpace
的選擇比較敏感,需要仔細(xì)調(diào)節(jié)以達(dá)到最佳效果。 - ?? 細(xì)節(jié)丟失風(fēng)險:如果參數(shù)設(shè)置不當(dāng)(尤其是
sigmaColor
過大),可能會導(dǎo)致一些細(xì)微的紋理細(xì)節(jié)被當(dāng)作噪聲而平滑掉。
結(jié)論
雙邊濾波是一種強(qiáng)大且實用的圖像平滑技術(shù),特別適用于那些既要降噪又希望保留清晰邊緣的應(yīng)用場景。通過理解其原理和 OpenCV 中 cv::bilateralFilter
函數(shù)的參數(shù),并進(jìn)行適當(dāng)?shù)膮?shù)調(diào)整,你可以有效地提升圖像質(zhì)量。希望本文能幫助你更好地理解和應(yīng)用雙邊濾波!
到此這篇關(guān)于c/c++中opencv雙邊濾波的實現(xiàn)的文章就介紹到這了,更多相關(guān)opencv 雙邊濾波內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語言中斐波那契數(shù)列的三種實現(xiàn)方式(遞歸、循環(huán)、矩陣)
本文主要介紹了C語言中斐波那契數(shù)列的三種實現(xiàn)方式(遞歸、循環(huán)、矩陣),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-01-01C++前綴樹字典樹的學(xué)習(xí)與模擬實現(xiàn)代碼示例
這篇文章主要介紹了C++前綴樹字典樹的學(xué)習(xí)與模擬實現(xiàn)代碼示例,Trie又被稱為前綴樹、字典樹,所以當(dāng)然是一棵樹,上面這棵Trie樹包含的字符串集合是{in,inn,int,tea,ten,to},每個節(jié)點的編號是我們?yōu)榱嗣枋龇奖慵由先サ?需要的朋友可以參考下2023-07-07