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