Обратные вызовы из C ++ обратно в Java - PullRequest
6 голосов
/ 17 июня 2011

Я кодировал следующее для численного интегрирования в C ++:

// integrate.h:
#ifdef BUILDING_DLL
#define  DLL_MACRO __declspec(dllexport)
#else
#define  DLL_MACRO __declspec(dllimport)
#endif

extern "C" {
  typedef double (*Function1VariablePtr)(double x);
  double DLL_MACRO integrate(Function1VariablePtr function, double min, double max);
}

// integrate.cpp: 
#include "integrate.h"
double integrate(Function1VariablePtr function, double min, double max) {
  const int n = 1001;
  double dx  = (max - min)/(1.0*(n - 1));
  double sum = 0.0;
  for(int i = 0; i < n; i++) {
    double xmid = min + (i + 0.5)*dx;
    sum += function(xmid)*dx;
  }
  return sum;
}

Теперь я хочу вызвать эту функцию из Java. Я нашел, как я могу реализовать интеграцию непосредственно в коде "моста" JNI:

// C++ "bridge" code to from/to Java:  
JNIEXPORT jdouble JNICALL 
Java_IntegrateJApp_JIntegrate(JNIEnv *jnienv, jclass jc,    
                              jdouble xmin, jdouble xmax) {
  jmethodID mid = jnienv->GetStaticMethodID(jc, "Function1D","(D)D");
  if (mid == 0)
    return - 1.0;
  const int n = 1001;
  double dx  = (xmax - xmin)/(1.0*(n - 1));
  double sum = 0.0;
  for(int i = 0; i < n; i++) {
    double xmid = xmin + (i + 0.5)*dx;
    double f = jnienv->CallStaticDoubleMethod(jc, mid, xmid);
    sum += f*dx;
  }
  return sum;
}

// Java code calling "bridge":
class IntegrateJApp { 
  public static void main(String[] args) { 
      System.loadLibrary("JIntegrate");
      double I = JIntegrate(0.0, 2*Math.PI);
      System.out.println( Double.toString(I) ); 
  } 
  public static double Function1D(double x) {
      return Math.sin(x);
  }
  public static native double JIntegrate(double xmin, double xmax);
} 

Однако я не хочу реализовывать числовую интеграцию непосредственно в коде моста C ++, а скорее вызывать код в integrate.cpp.

Как мне это сделать? Функция integrate () внутри integrate.cpp требует указателя функции, которого у меня нет. Есть ли способ получить указатель на функцию внутри Java с помощью JNI?

Заранее спасибо за любые ответы!

Ответы [ 2 ]

3 голосов
/ 17 июня 2011

Вы должны создать DLL с вашим кодом C ++ и вызывать ее из JNI

Загрузить DLL:

System.loadLibrary("PATH\\yourdllname.dll");

Создание ссылки на вашу функцию

public static native integrate(parameters);
2 голосов
/ 17 июня 2011

Один из способов сделать это - использовать указатель на функцию-член и изменить сигнатуру функции интегрирования.

См. Ниже общую идею:

functionwrapper.h

Объявите класс-оболочку функции.

class FunctionWrapper
{
public:
    typedef double (FunctionWrapper::*Function1VariablePtr)(double x);

    FunctionWrapper(JNIEnv*, jclass);
    double compute(double x);
};

integrate.h

Удалите предыдущий указатель на функцию typedef и измените методподпись для включения объекта-оболочки и указатель на его функцию-член.

#include "functionwrapper.h"
extern "C" {
    double DLL_MACRO integrate(FunctionWrapper*, FunctionWrapper::Function1VariablePtr, double min, double max);
}

integrate.cpp

Измените вызов функции на вызов функции-члена.

#include "integrate.h"
double integrate(FunctionWrapper* wrapper, FunctionWrapper::Function1VariablePtr function, double min, double max)
{
    // ...
    sum += (wrapper->*function)(xmid)*dx;
    // ...
}

возврат суммы;}

Код "моста" JNI:

Определите код оболочки и определите функцию, которая выполняет фактический вызов.Вызовите функцию integrate непосредственно из функции JNI:

#include "functionwrapper.h"

FunctionWrapper::FunctionWrapper(JNIEnv *jnienv, jclass jc) : m_jnienv(jnienv), m_jc(jc) {
    m_method= jnienv->GetStaticMethodID(jc, "Function1D","(D)D");
}

double FunctionWrapper:compute(double x) {
    return m_jnienv->CallStaticDoubleMethod(m_jc, m_method, x);;
}

// C++ "bridge" code to from/to Java:
JNIEXPORT jdouble JNICALL
Java_IntegrateJApp_JIntegrate(JNIEnv *jnienv, jclass jc,
                              jdouble xmin, jdouble xmax) {
    FunctionWrapper wrapper(jnienv, jc);
    return integrate(&wrapper, &FunctionWrapper::compute, 2, 3);
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...