1

JNI を使用して simulink Matlab から Java メソッドを呼び出そうとしています。最初のテストとして画面に helloworld のみを出力する main メソッドを呼び出す C++ の小さなコードを開発しましたが、クラスを見つけるために呼び出す行で matlab がクラッシュします。C++ コードは次のとおりです。

#include <stdio.h>
#include <jni.h>
#include "mex.h"
class MatlabAmbassador {

public:
    MatlabAmbassador ();
    //Destructor
   ~MatlabAmbassador ();
    void run();
      JNIEnv* create_vm() ;
    void invoke_class(JNIEnv* env);   
private:

}; // end class

MatlabAmbassador::MatlabAmbassador() {

}

MatlabAmbassador::~MatlabAmbassador() {

}


// -------------------------------------------------------------------------

JNIEnv* MatlabAmbassador::create_vm() {
    JavaVM* jvm;
JNIEnv* env;
JavaVMInitArgs args;
JavaVMOption options[1];
long status;
/* There is a new JNI_VERSION_1_4, but it doesn't add anything for the purposes of our example. */
args.version = JNI_VERSION_1_6;
args.nOptions = 1;
    options[0].optionString = "-Djava.class.path=C:\\Apps\\Projetos em java\\Portico\\Portico_Agent\\bin";
args.options = options;
args.ignoreUnrecognized = JNI_FALSE;

 status=JNI_CreateJavaVM(&jvm, (void **)&env, &args);
return env;
}
// -------------------------------------------------------------------------

void MatlabAmbassador::invoke_class(JNIEnv* env) {
jclass helloWorldClass;
jmethodID mainMethod;
jobjectArray applicationArgs;
jstring applicationArg0;

  mexPrintf("First\n");
    helloWorldClass = env->FindClass("Teste");  <--- MATLAB CRASHES HERE
    mexPrintf("second\n");
    if (env->ExceptionOccurred()) { 
 env->ExceptionDescribe(); 
 env->ExceptionClear() ;
 } 
     if ( helloWorldClass == NULL ) {
        mexPrintf( "%s%s\n", "Unable to obtain class reference for ", 
 helloWorldClass );
        return;
    } else {
        mexPrintf( "%s%s\n", "Sucessfully created class reference for ", 
helloWorldClass );
    }

    mainMethod = env->GetStaticMethodID( helloWorldClass, "main", "([Ljava/lang/String;)V");

    applicationArgs = env->NewObjectArray(1, env->FindClass("java/lang/String"), NULL);
    applicationArg0 = env->NewStringUTF( "From-C-program");
    env->SetObjectArrayElement( applicationArgs, 0, applicationArg0);

    env->CallStaticVoidMethod( helloWorldClass, mainMethod, applicationArgs); 
}

// -------------------------------------------------------------------------
void MatlabAmbassador::run() {

 char str [80];

 mexPrintf(" INITIALIZING....\n" );
    JNIEnv* env = create_vm();
    invoke_class( env );


}
 // -------------------------------------------------------------------------

Simulink Matlab には、使用するメソッドがいくつかあります。このメソッドの 1 つは、上記のメソッド run() を呼び出すために使用されます。以下にコードを示します。

static void mdlStart(SimStruct *S)
{
    char *buf;
    size_t buflen;
    int status;

   buflen = mxGetN((ssGetSFcnParam(S, 2)))*sizeof(mxChar)+1 ; // le o 3o parametro passado pela     funcao (nome do arq)
   buf = (char *)mxMalloc(buflen); // aloca memoria
   status = mxGetString((ssGetSFcnParam(S, 2)), buf,(mwSize)buflen);

 ssGetPWork(S)[0] = (void *) new MatlabAmbassador; // store new C++ object in the

MatlabAmbassador *c = (MatlabAmbassador *) ssGetPWork(S)[0];

c->run();

単純な Java コードは次のとおりです。

public class Teste 
{
  public static void main(String[] args)
  {

      System.out.println(" INITALIZING...");


  }
}

だから、私が欠けているものを説明したり、Matlab内でJNIを呼び出すのに実際の問題があるかどうかを説明したりできますか. Matlab はバージョン 2011b で、インストールされている Java のバージョンは JDK 1.0.6_45 です。

できるだけ早く助けていただければ幸いです。

宜しくお願いします アンドレ・ヌーデル andre.nudel@gmail.com

4

2 に答える 2

0

私は Simulink にあまり詳しくないので、代わりに通常の MEX 関数の例を示しています。

@AndrewJankeの提案に従って、新しい JVM インスタンスを作成する代わりに、MATLAB プロセス内で実行されている既存の JVM インスタンスを取得します。

jni.cpp

#include "mex.h"
#include "jni.h"

void mexFunction(int nlhs, mxArray *plhs[], int nrhs, const mxArray *prhs[])
{
    // get MATLAB Java virtual machine instance
    JavaVM *vm = NULL;
    int n; jint res;
    res = JNI_GetCreatedJavaVMs(&vm, 1, (jsize*) &n);
    if (res != JNI_OK || n != 1) {
        mexErrMsgIdAndTxt("mex:jni", "Couldn't find existing Java VM");
    }

    // get JNI interface instance
    JNIEnv *env = NULL;
    res = vm->GetEnv((void**) &env, JNI_VERSION_1_6);
    if (res != JNI_OK || env == NULL) {
        mexErrMsgIdAndTxt("mex:jni", "Couldn't get Java JNI environment");
    }

    mexPrintf("using Java %d.%d\n",
        env->GetVersion() >> 16, env->GetVersion() & 0xFFFF);
}

以下を使用してファイルをコンパイルしました。

mex -I"C:\Program Files\Java\jdk1.6.0_45\include" 
    -I"C:\Program Files\Java\jdk1.6.0_45\include\win32" jni.cpp
    "C:\Program Files\Java\jdk1.6.0_45\lib\jvm.lib"

MATLAB で実行すると、次のようになります。

>> jni
using Java 1.6

上記の例を拡張して、外部 Java クラスを呼び出すことができるはずです。MATLAB MEX-API と Java JNI API の間でデータを渡すことができなかったことに注意してください (つまり、Java オブジェクトをそのまま MEX 関数に渡したり、逆に何らかの種類のものをjobjectそのまま直接渡したりすることを意味します)。変換せずに MATLAB に変換します)。

>> x = java.lang.Double(1)
>> my_mex_jni_fcn(x)

問題はそれmxArrayが不透明な型であるため、返されたポインターを解釈する方法がわからないことですmxGetData

于 2013-07-02T21:22:59.220 に答える
0

確実にJNI_CreateJavaVM成功しますか?あなたのコードは、返されたステータス コードまたは結果のenv値をチェックしていません (初期化されていないため、ガベージが含まれている可能性があります) env

これを Matlab 内で実行している場合、Matlab の通常の環境の一部としてプロセス内で JVM が既に実行されているため、JVM の作成が失敗する可能性があります。JNI ドキュメントには、「1 つのプロセスでの複数の VM の作成はサポートされていません」と記載されています ( http://docs.oracle.com/javase/6/docs/technotes/guides/jni/spec/invocation.html#wp636 )。これが起こっている場合は、 (-5)JNI_CreateJavaVMを返しJNI_EEXISTます。ここで起こっていることではない場合でも、失敗する可能性のある関数から返されるステータス コードを確認することをお勧めします。JNI ドキュメントの例では、「わかりやすくするために」エラー チェックを省略していますが、実際に実行するコードにはエラー チェックを含める必要があります。

によって返されたステータスを確認し、JNI_CreateJavaVMそれを出力して、成功していることを確認します。また、おそらくenv0 に初期化して、JNI からポインターを取得したのか、単にランダム データを取得したのかを明確にします。

于 2013-07-01T04:00:00.740 に答える