OpenCV - сопоставление объектов с использованием дескрипторов SURF и BruteForceMatcher - PullRequest
19 голосов
/ 04 сентября 2011

У меня есть вопрос об объектах, соответствующих OpenCV. Я использую алгоритм SURF, реализованный в opencv 2.3, чтобы сначала обнаружить функции на каждом изображении, а затем извлекать дескрипторы этих функций. Проблема в сопоставлении с использованием Brute Force Matcher, я не знаю, как я могу судить, что два изображения совпали или нет, потому что когда я использую два разных изображения, между дескрипторами на этих двух изображениях есть линии!

Эти выходные данные моего кода, либо два изображения - я сравниваю с ними - похожи или различаются, результирующее изображение указывает на то, что два изображения совпадают.

Вопрос в том, как различить два изображения?

Истинное соответствие:

http://store1.up-00.com/Jun11/hxM00286.jpg

Ложное совпадение !! :

http://store1.up-00.com/Jun11/D5H00286.jpg

Мой код:

Mat image1, outImg1, image2, outImg2;

// vector of keypoints
vector<KeyPoint> keypoints1, keypoints2;

// Read input images
image1 = imread("C://Google-Logo.jpg",0);
image2 = imread("C://Alex_Eng.jpg",0);

SurfFeatureDetector surf(2500);
surf.detect(image1, keypoints1);
surf.detect(image2, keypoints2);
drawKeypoints(image1, keypoints1, outImg1, Scalar(255,255,255), DrawMatchesFlags::DRAW_RICH_KEYPOINTS);
drawKeypoints(image2, keypoints2, outImg2, Scalar(255,255,255), DrawMatchesFlags::DRAW_RICH_KEYPOINTS);

namedWindow("SURF detector img1");
imshow("SURF detector img1", outImg1);

namedWindow("SURF detector img2");
imshow("SURF detector img2", outImg2);

SurfDescriptorExtractor surfDesc;
Mat descriptors1, descriptors2;
surfDesc.compute(image1, keypoints1, descriptors1);
surfDesc.compute(image2, keypoints2, descriptors2);

BruteForceMatcher<L2<float>> matcher;
vector<DMatch> matches;
matcher.match(descriptors1,descriptors2, matches);

nth_element(matches.begin(), matches.begin()+24, matches.end());
matches.erase(matches.begin()+25, matches.end());

Mat imageMatches;
drawMatches(image1, keypoints1, image2, keypoints2, matches, imageMatches, Scalar(255,255,255));

namedWindow("Matched");
imshow("Matched", imageMatches);

cv::waitKey();
return 0;

Ответы [ 3 ]

19 голосов
/ 05 сентября 2011

Проблема заключалась в использовании только Brute Force Matcher, я нашел методы для получения набора хороших совпадений между двумя представлениями в "Руководство по программированию приложений OpenCV 2 Computer Vision"

Ch9: сопоставление изображений с использованием случайной выборки

Они используют K-Nearest Neighbor и RANSAC

И спасибо

9 голосов
/ 19 декабря 2012

Для удаления выбросов RANSAC + гомография - это хороший метод для сравнения двух плоских изображений.

Гомография - это модель, которую RANSAC попытается использовать для сравнения точек на обоих изображениях.найти наилучший набор точек, которые лучше соответствуют проекционной модели гомографии (преобразование из одной плоскости в другую).

cv::findHomography(srcPoints,dstPoints, RANSAC, status);

Функция выше вернет статус массива, который имеет 1 для индексов, рассматриваемых как внутренниеи 0 для индексов, которые считаются выбросами, поэтому вы можете удалить выбросы, проверив этот массив состояний.

4 голосов
/ 01 марта 2013

Вам нужно изменить свой гессиан, 2500 слишком много.Попробуйте 50. Когда вы используете большой гессиан, в результате получается много ключевых точек, что приводит к ненужным результатам.Другая информация о SURF заключается в том, что ваш маркер должен быть более насыщенным и содержать больше деталей.

...