Я пытаюсь реализовать свой собственный фильтр NL-средств, используя библиотеку opencv. Но после того, как я попытался использовать этот фильтр, я получаю черные пиксели на результирующем изображении.
double gaussian(float x, double sigma)
{
return (1 / (2 * M_PI*sigma*sigma))*(exp(-(x*x) / (2 * sigma*sigma)));
}
void applyNLMFilter (Mat source, Mat filteredImage, int x, int y, int diameter, double sigmaS,int searchwindow)
{
int i1, j1;
int min_sad;
int half = diameter /2 ;
int min_sad1 = 1000000;
Mat originblock = source(Rect(y, x, half, half));
for (i1 = -searchwindow * half + x * half; i1 < searchwindow * half + x * half; i1++)
for (j1 = -searchwindow * searchwindow + y * half; j1 < searchwindow * half + y * half; j1++)
{
if ((i1 < 0) || (j1 < 0) || (i1 > source.cols - half) || (j1 > source.rows - half)) continue;
else {
Mat blcktocmp = source(Rect(i1, j1, half, half));
Mat diff;
cv::absdiff(originblock, blcktocmp, diff);
double s = cv::sum(diff)[0];
int sad = cv::sum(diff)[0];
if (sad <= min_sad1)
{
min_sad = sad;
}
}
}
double iFilteredb = 0, iFilteredg = 0, iFilteredr = 0;
double wP = 0;
int neighbor_x = 0;
int neighbor_y = 0;
for (int i = 0; i < diameter; i++) {
for (int j = 0; j < diameter; j++) {
neighbor_x = x - (half - i);
neighbor_y = y - (half - j);
Vec3b neighborPixel = source.at<Vec3b>(neighbor_x, neighbor_y), centerPixel = source.at<Vec3b>(x, y);
double gs = gaussian(min_sad, sigmaS);
double w = gs;
iFilteredb = iFilteredb + neighborPixel.val[0] * w;
iFilteredg = iFilteredg + neighborPixel.val[1] * w;
iFilteredr = iFilteredr + neighborPixel.val[2] * w;
wP = wP + w;
}
}
iFilteredb = iFilteredb / wP;
iFilteredg = iFilteredg / wP;
iFilteredr = iFilteredr / wP;
Vec3b filtered;
filtered.val[0] = iFilteredb;
filtered.val[1] = iFilteredg;
filtered.val[2] = iFilteredr;
filteredImage.at<Vec3b>(x, y) = filtered;
}Mat NLMfilter(Mat source, int diameter, double sigmaS, int searchwindow)
{
Mat filteredImage = source.clone();
int width = source.cols;
int height = source.rows;
int half = diameter / 2;
for (int i = half; i < height - diameter; i++) {
for (int j = half; j < width - diameter; j++) {
applyNLMFilter(source, filteredImage, i, j, diameter, sigmaS, searchwindow);
}
}
return filteredImage;
}
И вот результат:
Исходное изображение
Отфильтрованное изображение
Ход работы — для каждого пикселя (кроме тех, что на границе) я ищу наиболее похожий патч, используя сумму абсолютных разностей. Затем я использую эту сумму для вычисления функции Гаусса, применяя ее к значению пикселя и вычисляя значение результата для отфильтрованного пикселя.
Где я ошибся?
Задача ещё не решена.
Других решений пока нет …