您有三个相关的问题:
1/ 如何在iOS项目中运行openCV框架
2/ 如何在iOS项目中运行模板匹配c++示例代码
3/ 如何使用相机视图进行实时模板匹配
1/ 如何在iOS项目中运行openCV框架
- 按照您所描述的下载并导入openCV框架。
- 按照您所描述的更改.pch文件。
检查您的目标构建设置中是否将c++标准库设置为libc++(这是新项目的默认设置)。
不要只是导入demo.cpp而不进行下面所述的更改(它是一个“原始”的c++程序,具有自己的main
函数,需要进行修改才能作为iOS / Cocoa项目的一部分工作)。
不要搞乱头文件搜索路径、其他链接器标志等,如果您已经从openCV.org导入了预构建的框架,这是不必要的。
不要将您的.m文件更改为.mm,除非您知道需要这样做。我的建议是尽可能地将您的c++代码与您的objective-C代码分开,因此大多数文件应该是.m文件(objective-C)或.cpp文件(c++)。只有在您打算在同一文件中混合objective-C和c++时,才需要.mm前缀。
2/ 如何在iOS项目中运行模板匹配c++示例代码
我们将设置这样一个机制,使得您的iOS viewController - 以及大部分iOS代码 - 不需要知道图像是使用openCV / c++处理的,同样的,C++代码也不需要知道它的输入或输出图像数据被路由到哪里。我们通过在两者之间创建一个小型包装器类来实现此目的,该类将objective-C方法调用转换为c++类成员函数并返回。我们还将在UIImage上设置一个类别,以将图像格式从iOS友好的UIImage转换为openCV本地的cv::Mat。
UIImage+OpenCV类别
您需要一些实用程序方法来将UIImage转换为cv::Mat并返回。将它们放置在UIImage类别中是一个好的选择。在XCode中:文件>新建文件>Cocoa Touch>Objective-C类别将为您设置。将该类别命名为OpenCV,并将其设置为UIImage的类别。您需要将此.m文件更改为.mm,因为它需要理解来自openCV框架的c++类型。
头文件应如下所示:
#import <UIKit/UIKit.h>
@interface UIImage (OpenCV)
+ (UIImage *)imageWithCVMat:(const cv::Mat&)cvMat;
- (cv::Mat)cvMat;
@end
.mm文件应通过紧密遵循此openCV.org代码示例实现这些方法,以作为类别方法进行调整(例如,您不会将UIImage传递到实例方法中,而是使用self
引用它)。
您可以像这样使用类别方法,就好像它们是UIImage类和实例方法:
UIImage* image = [UIImage imageWithCVMat:matImage]; //class method
cv::Mat matImage = [image cvMat]; //instance method
openCV包装类
创建一个包装类,将您从viewController调用的Objective-C方法转换为C++函数
头文件应该像这样
#import <Foundation/Foundation.h>
@interface CVWrapper : NSObject
+ (NSImage*) templateMatchImage:(UIImage*)image
patch:(UIImage*)patch
method:(int)method;
@end
我们发送模板图像、路径图像和模板匹配方法,并获得一张显示匹配结果的图像。
实现(.mm文件)
#import "CVWrapper.h"
#import "CVTemplateMatch.h"
#import "UIImage+OpenCV.h"
@implementation CVWrapper
+ (UIImage*) templateMatchImage:(UIImage *)image
patch:(UIImage *)patch
method:(int)method
{
cv::Mat imageMat = [image cvMat];
cv::Mat patchMat = [patch cvMat];
cv::Mat matchImage =
CVTemplateMatch::matchImage(imageMat,
patchMat,
method);
UIImage* result = [UIImage imageWithCVMat:matchImage];
return result;
}
我们实际上是将标准的Objective-C方法和UIImage类型转换为调用具有C++(openCV框架)类型的成员函数,并将结果转换回UIImage。
C++ TemplateMatch类
头文件:
#ifndef __CVOpenTemplate__CVTemplateMatch__
#define __CVOpenTemplate__CVTemplateMatch__
class CVTemplateMatch
{
public:
static cv::Mat matchImage (cv::Mat imageMat,
cv::Mat patchMat,
int method);
};
#endif
@end
实现:
这是将模板匹配openCV示例代码重新设计为类实现的代码:
#include "CVTemplateMatch.h"
#include "opencv2/imgproc/imgproc.hpp"
#include <iostream>
#include <stdio.h>
using namespace std;
using namespace cv;
Mat img; Mat templ; Mat result;
int match_method;
Mat MatchingMethod( int, void* );
Mat CVTemplateMatch::matchImage (Mat image,Mat patch, int method)
{
img = image;
templ = patch;
match_method = method;
Mat result = MatchingMethod( 0, 0 );
return result;
}
Mat MatchingMethod( int, void* )
{
Mat img_display;
img.copyTo( img_display );
int result_cols = img.cols - templ.cols + 1;
int result_rows = img.rows - templ.rows + 1;
result.create( result_cols, result_rows, CV_32FC1 );
matchTemplate( img, templ, result, match_method );
normalize( result, result, 0, 1, NORM_MINMAX, -1, Mat() );
double minVal; double maxVal; Point minLoc; Point maxLoc;
Point matchLoc;
minMaxLoc( result, &minVal, &maxVal, &minLoc, &maxLoc, Mat() );
if( match_method == CV_TM_SQDIFF || match_method == CV_TM_SQDIFF_NORMED )
{ matchLoc = minLoc; }
else
{ matchLoc = maxLoc; }
rectangle( img_display, matchLoc, Point( matchLoc.x + templ.cols , matchLoc.y + templ.rows ), Scalar::all(0), 2, 8, 0 );
rectangle( result, matchLoc, Point( matchLoc.x + templ.cols , matchLoc.y + templ.rows ), Scalar::all(0), 2, 8, 0 );
return img_display;
}
现在在你的
viewController
中,你只需要调用这个方法:
UIImage* matchedImage =
[CVWrapper templateMatchImage:self.imageView.image
patch:self.patchView.image
method:0]
没有C++,不用担心。
3/ 通过实时摄像头视图进行模板匹配
简单来说:matchTemplate
在实时摄像头的情况下效果不好。该算法在图像中寻找与补丁相同比例和方向的匹配项:它将补丁瓷砖沿其原始方向和大小滑动到图像上,进行最佳匹配比较。如果图像被透视扭曲、大小不同或旋转到不同的方向,则这样做将不能产生很好的结果。
您可以考虑使用OpenCV的特征检测算法,其中一些已经移至非免费版。这里是SIFT的一个很好的描述,可以给您提供思路。对于视频捕获,您可能还需要查看opencv2/highgui
中的cap_ios.h
:这里有一个教程。