与cvFindContours一起使用cvCalcPGH

时间:2012-08-08 06:48:27

标签: opencv

在某些原因,谷歌没有使用cvCalcPGH()的好例子,也没有使用Freeman代码进行轮廓匹配的其他好例子。所以我试着去做:

#include "opencv2/opencv.hpp"
#include "opencv2/legacy/legacy.hpp"

int
main(int argc, char* argv[]) {

        IplImage* g_gray = cvLoadImage("lisa.png",CV_LOAD_IMAGE_GRAYSCALE);
        cvThreshold( g_gray, g_gray, 100, 255, CV_THRESH_BINARY );

        CvMemStorage* storage=cvCreateMemStorage(0);
        CvSeq* firstContour=NULL;

        int a=cvFindContours(g_gray, storage, &firstContour,sizeof(CvChain),CV_RETR_LIST, CV_CHAIN_CODE);

        CvHistogram *hist;
        int h_bins = 30, s_bins = 30;
        float h_ranges[] = { 0, 180 };
        float s_ranges[] = { 0, 255 };
        int hist_size[] = { h_bins, s_bins };
        float* ranges[] = { h_ranges, s_ranges };

        hist = cvCreateHist( 2, hist_size, CV_HIST_ARRAY, ranges, 1 );
        if( CV_SEQ_ELTYPE( firstContour ) != CV_32SC2 ) {
                printf(":(\n");
        }

        cvCalcPGH((CvSeq *)firstContour,hist);
        return 0;

}

cvFindContours()将按文档运行,但cvCalcPGH()将失败:

$ ./calcpgh 
:(
OpenCV Error: Unsupported format or combination of formats (The contour is not valid or the point type is not supported) in cvCalcPGH, file /Users/tonu-samuel/OpenCV-2.4.2/modules/legacy/src/pgh.cpp, line 351
terminate called throwing an exceptionAbort trap: 6
$

OpenCV是2.4.2并且表示assert()用于CV_SEQ_ELTYPE( contour ) != CV_32SC2我也检查了我的代码。如何解决它不确定。

1 个答案:

答案 0 :(得分:6)

谷歌不会给你很好的例子,你所谈论的所有内容都已被弃用。

新的opencv中不再提供

cvCalcPGH,您可能已经知道,因为您已经包含了旧版库。

此外CV_CHAIN_CODEremoved,您无法在新的opencv中使用它。

你应该找到时间将你的代码移植到新的opencv,这是一个趋势,并将从其他人那里获得更多的帮助。

但是讨论你当前的代码,正如旧文档所说:

  

CV_CHAIN_CODE - 输出Freeman链码中的轮廓。所有其他   方法输出多边形(顶点序列)

cvFindContours的输出始终是顶点序列,而CV_CHAIN_CODE只是一个例外。这就是cvCalcPGH无法理解这些代码的原因。

由于您可能已经使用其他近似方法测试了代码,您应该已经看到cvCalcPGH可以正常使用多边形。

你应该自己将Freeman代码转换为多边形,然后你可以使用你想要的任何opencv方法。但如果没有这个,我还没有看到可以解释Freeman代码本身的单一方法。

我不确定应该如何完成转换,但是使用下面的代码,从Sara Maher Mohammad无耻地窃取,您可以提取代码并开始自己解释:

CvChainPtReader reader;

cvStartReadChainPoints((CvChain*) firstContour, &reader);
for (int i = 0; i < firstContour->total; i++) {
    CV_READ_SEQ_ELEM(reader.code, (*((CvSeqReader*)&(reader))));
    printf("%d \n", reader.code);
}

但同样,我不建议处理弃用的方法和功能,你可能会找到更好的方法来做你想做的事。

只是猜测你可能只是想找到两个Freeman代码之间的距离,你可能只是不计算PGH,只是使用其他一些距离函数,如所讨论的here