2013-04-20 92 views
0

目前,我正在试图封装自定义C++库以用于Java(并最终使用Android)。我的图书馆使用OpenCV MatSize类作为输入。下面,是我到目前为止的例子痛饮接口:使用SWIG引用外部库中的特定类

%module customLib 

%include <std_vector.i> 
%include <std_string.i> 

%{ 
#include "opencv2/core/core.hpp" 
#include "../include/myinc/CustomLib.h" 
%} 

namespace cv { 
    class Mat {}; 
    class Size {}; 
} 

namespace myinc { 
    CustomType preprocessMatrix(const cv::Mat& src, cv::Mat& dst, const cv::Size& ksize); 
} 

OpenCV的为MatSize已经Java包装;我想知道我在这里做了什么会导致命名冲突,一旦我添加OpenCV自己的Java框架。这是通过SWIG引用外部库类的正确方法吗?或者,有没有更好的方法来完成这种类型的事情?

回答

1

我终于明白了这一点!我花了些时间了解SWIG的Java类型映射系统是如何工作的(大多数例子都掩盖了一些重要的概念)。

如果其他人在理解如何编写Java类型映射时遇到困难,这是一种帮助我的方法。我认为从上到下的角度(即Java - >中间JNI - > JNI)查看类型映射是最容易的。首先,将jstype定义为您希望在Java类中看到的类型(例如,module_name.java)。使用javain告诉SWIG如何将变量从Java类传递给中间JNI类(例如,module_nameJNI.java)。接下来,将jtype定义为与javain相同的类型。例如,$javainput.getNativeObjectAddr()返回long,所以这将成为我的jtype。最后,将jni定义为实际的JNI函数将使用的内容。

下面,是SWIG typemaps我想出了OpenCV的上的Java接口上运行(如2.4.5):

%typemap(jstype) cv::Mat& "org.opencv.core.Mat" 
    %typemap(javain) cv::Mat& "$javainput.getNativeObjAddr()" 
    %typemap(jtype) cv::Mat& "long" 
    %typemap(jni) cv::Mat& "jlong" 

    %typemap(in) cv::Mat& { 
      $1 = *(cv::Mat **)&$input; 
    } 

    %typemap(jstype) cv::Size& "org.opencv.core.Size" 
    %typemap(javain) cv::Size& "$javainput" 
    %typemap(jtype) cv::Size& "org.opencv.core.Size" 
    %typemap(jni) cv::Size& "jobject" 

    %typemap(in) cv::Size& { 
      jclass sizeClass = JCALL1(GetObjectClass, jenv, $input); 
      jfieldID widthFieldId = JCALL3(GetFieldID, jenv, sizeClass, "width", "D"); 
      jfieldID heightFieldId = JCALL3(GetFieldID, jenv, sizeClass, "height", "D"); 
      double width = JCALL2(GetDoubleField, jenv, $input, widthFieldId); 
      double height = JCALL2(GetDoubleField, jenv, $input, heightFieldId); 
      $1 = new cv::Size((int)width, (int)height); 
    } 

    %typemap(freearg) cv::Size& { 
      delete $1; 
    }