Подтвердить что ты не робот

Код примера OpenCV для поиска контуров: проблема освобождения вектора

Я пытаюсь начать с определения контура в OpenCV 2.4.2. С этой целью я создал проект для OpenCV и скопировал весь код примера, найденный в документации . Для дальнейшего использования здесь приведен код:

#include "opencv2/highgui/highgui.hpp"
#include "opencv2/imgproc/imgproc.hpp"
#include <iostream>
#include <stdio.h>
#include <stdlib.h>

using namespace cv;
using namespace std;

Mat src; Mat src_gray;
int thresh = 100;
int max_thresh = 255;
RNG rng(12345);

/// Function header
void thresh_callback(int, void* );

/** @function main */
int main( int argc, char** argv )
{
  /// Load source image and convert it to gray
  src = imread( argv[1], 1 );

  /// Convert image to gray and blur it
  cvtColor( src, src_gray, CV_BGR2GRAY );
  blur( src_gray, src_gray, Size(3,3) );

  /// Create Window
  char* source_window = "Source";
  namedWindow( source_window, CV_WINDOW_AUTOSIZE );
  imshow( source_window, src );

  createTrackbar( " Canny thresh:", "Source", &thresh, max_thresh, thresh_callback );
  thresh_callback( 0, 0 );

  waitKey(0);
  return(0);
}

/** @function thresh_callback */
void thresh_callback(int, void* )
{
  Mat canny_output;
  vector<vector<Point> > contours;
  vector<Vec4i> hierarchy;

  /// Detect edges using canny
  Canny( src_gray, canny_output, thresh, thresh*2, 3 );
  /// Find contours
  findContours( canny_output, contours, hierarchy, CV_RETR_TREE, CV_CHAIN_APPROX_SIMPLE, Point(0, 0) );

  /// Draw contours
  Mat drawing = Mat::zeros( canny_output.size(), CV_8UC3 );
  for( int i = 0; i< contours.size(); i++ )
     {
       Scalar color = Scalar( rng.uniform(0, 255), rng.uniform(0,255), rng.uniform(0,255) );
       drawContours( drawing, contours, i, color, 2, 8, hierarchy, 0, Point() );
     }

  /// Show in a window
  namedWindow( "Contours", CV_WINDOW_AUTOSIZE );
  imshow( "Contours", drawing );

  contours.clear(); // Error!!
}

Он отлично компилируется в Visual Studio 11 RC (Windows 7 SP1), но я получаю сообщение об ошибке в конце функции thresh_callback. Вот stacktrace:

msvcr110d.dll!_CrtIsValidHeapPointer(const void * pUserData) Line 2036
msvcr110d.dll!_free_dbg_nolock(void * pUserData, int nBlockUse) Line 1322
msvcr110d.dll!_free_dbg(void * pUserData, int nBlockUse) Line 1265
msvcr110d.dll!operator delete(void * pUserData) Line 54
std::allocator<cv::Point_<int> >::deallocate(cv::Point_<int> * _Ptr, unsigned int __formal) Line 586
std::_Wrap_alloc<std::allocator<cv::Point_<int> > >::deallocate(cv::Point_<int> * _Ptr, unsigned int _Count) Line 888
std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > >::_Tidy() Line 1542
std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > >::~vector<cv::Point_<int>,std::allocator<cv::Point_<int> > >() Line 901
std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > >::`scalar deleting destructor'(unsigned int)
std::allocator<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > >::destroy<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > >(std::<cv::Point_<int>,std::allocator<cv::Point_<int> > > * _Ptr) Line 624
std::allocator_traits<std::allocator<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > > >::destroy<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > >(std::allocator<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > > & _Al, std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > * _Ptr)758
std::_Wrap_alloc<std::allocator<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > > >::destroy<std::vector<cv::Point_<int>,std::allocator<cv::Poin> > > >(std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > * _Ptr) Line 909
std::_Destroy_range<std::_Wrap_alloc<std::allocator<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > > > >(std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > * _First, std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > * _Last, std::_Wrap_alloc<std::allocator<std::vector<cv::Point_<int>,std::allocator<cv::<int> > > > > & _Al, std::_Nonscalar_ptr_iterator_tag __formal) Line 89
std::_Destroy_range<std::_Wrap_alloc<std::allocator<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > > > >(std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > * _First, std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > * _Last, std::_Wrap_alloc<std::allocator<std::vector<cv::Point_<int>,std::allocator<cv::<int> > > > > & _Al) Line 80
std::vector<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > >,std::allocator<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> >::_Destroy(std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > * _First, std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > * _Last) Line 1480
std::vector<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > >,std::allocator<std::vector<cv::Point_<int>,std::allocator<cv::Point_<int> > > > >::c Line 1416
thresh_callback(int __formal, void * __formal) Line 143
opencv_highgui242d.dll!icvUpdateTrackbar(CvTrackbar * trackbar, int pos) Line 1938
opencv_highgui242d.dll!HGToolbarProc(HWND__ * hwnd, unsigned int uMsg, unsigned int wParam, long lParam) Line 1982

(Обратите внимание, что на самом деле у меня есть слегка измененная версия кода примера, добавлено несколько запросов printf-s и contours.clear(), которое находится в строке 143, и которое запускает векторное освобождение [которое будет происходить автоматически при конец функции], который, как представляется, является источником проблемы. Сама же проблема возникает и с точным кодом примера.)

Проблема заключается в освобождении вектора contours. То же самое происходит, если я попытаюсь позвонить contours.clear(). Я изучил содержимое переменной в отладчике VS, но ничего необычного, насколько я могу судить, не было.

Я попытался очистить вектор, как только он больше не нужен (после цикла for), но это не помогло. Я также попытался переключить Platform Toolset в Visual Studio 10, который, в свою очередь, даже не будет компилироваться с сообщениями об ошибках, которые находятся вне меня:

error C1083: Cannot open include file: 'SDKDDKVer.h': No such file or directory (C:\<project path>\targetver.h) (Line 8, Column 1)
IntelliSense: cannot open source file "SDKDDKVer.h" (C:\<project path>\targetver.h) (Line 8, Column 1)
IntelliSense: cannot open source file "windows.h" (C:\<opencv path>\build\include\opencv2\core\operations.hpp (Line 83, Column 3)

Любая помощь будет принята с благодарностью. Обратите внимание: я не программист на С++: у меня очень мало знаний и даже меньше опыта программирования на С++ и встроенного программирования в целом.

Изменить: выясняется, что отладчик отображает неправильную строку как источник ошибки. Я должен был видеть в стеке вызовов, что проблема была с vector > . Итак, проблематичный вектор contours, а не hierarchy!

Изменить # 2. Я также попытался воспроизвести проблему с минимальным кодом, который создает vector<vector<cv::Point> >, помещает в него некоторые элементы, а затем очищает его, но я не смог воспроизвести проблему.

4b9b3361

Ответ 1

Вы создаете свое приложение в режиме отладки и привязываетесь к многопотоковому DLL CRT. Вы знаете, с каким CRT связаны DLL OpenCV? Если он связан с статическим CRT, он заполнит вектор данными, выделенными из отдельной кучи, что вызывает утверждение в используемом вами алгоритме Debug.

Если вы создадите приложение в режиме Release, вы больше не увидите утверждение, но вы можете потерять память. Лучше всего убедиться, что и ваше приложение, и DLL OpenCV связаны с одним и тем же многопотоковым DLL-CRT.

EDIT: Если вы не можете перестроить OpenCV, чтобы использовать тот же CRT, что и ваше приложение, вы можете попытаться сказать компоновщику использовать ту же версию CRT, что и OpenCV для вашего приложения, изменив манифест приложения. См. Как заставить компилятор С++ использовать конкретную версию CRT? для получения дополнительной информации о том, как это сделать.

Ответ 2

Перейдите в свои варианты решения, там параметры c-runtime (CRT) и проверьте связь (CRT, как указано выше)... если ваш проект был/был (один раз) создан с помощью VS10, и вы используете более новую версию, просто отредактируйте ссылки не 10, а 11 или 12..

Ответ 3

Добавить в системный путь (в качестве примера): H:\Source\opencv_v2_4_13_2\выход\Bin\Debug

где каталог: "H:\Source\opencv_v2_4_13_2\выход" используется из CMake в качестве целевого каталога для визуальных проектов Studio 14 (2015) opencv.

В вашем проекте используются файлы opencv dll, но не находит: opencv_core2413d.dll