Ho appena iniziato con JNI e ho un problema seguente.JNI mantenendo un riferimento globale a un oggetto, accedendo ad esso con altri metodi JNI. Mantenere vivo un oggetto C++ su più chiamate JNI
Ho una libreria C++ che ha una classe semplice. Ho tre metodi JNI chiamati dal progetto Java Android che, instatiate la classe, chiamano un metodo sulla classe istanziata e lo distruggono, rispettivamente. Mantengo un riferimento globale a questo oggetto, quindi sarebbe disponibile per me negli altri due metodi JNI.
Sospetto di non poterlo fare. Quando eseguo l'app, ottengo un errore di runtime (riferimento usato stantio), e sospetto che ciò sia dovuto al fatto che la referenza globale non è valida alle chiamate successive ad altri metodi JNI.
È l'unico modo per ottenere ciò che voglio (avere l'oggetto in diretta su più chiamate JNI), in realtà passare il puntatore alla classe istanziata su Java, tenerlo in giro e quindi passarlo di nuovo al Funzioni JNI? Se è così, va bene, voglio essere sicuro di non poterlo fare con un riferimento globale, e non mi manca solo qualcosa.
Ho letto la documentazione ei capitoli relativi ai riferimenti globali/locali in JNI, ma sembra che ciò si applichi solo alle classi Java e non alle mie classi native C++ o mi sbaglio.
Ecco il codice se la mia descrizione non è chiara (riassumendo, mi chiedo se questo meccanismo di persistenza degli oggetti funziona affatto):
Java:
package com.test.ndktest;
import android.app.Activity;
import android.os.Bundle;
import android.app.AlertDialog;
public class NDKTestActivity extends Activity {
static {
System.loadLibrary("ndkDTP");
}
private native void initializeTestClass();
private native void destroyTestClass();
private native String invokeNativeFunction();
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
initializeTestClass();
String hello = invokeNativeFunction();
destroyTestClass();
new AlertDialog.Builder(this).setMessage(hello).show();
}
}
JNI intestazione:
extern "C" {
jstring Java_com_test_ndktest_NDKTestActivity_initializeTestClass(JNIEnv* env, jobject javaThis);
jstring Java_com_test_ndktest_NDKTestActivity_destroyTestClass(JNIEnv* env, jobject javaThis);
jstring Java_com_test_ndktest_NDKTestActivity_invokeNativeFunction(JNIEnv* env, jobject javaThis);
};
JNI corpo:
#include <string.h>
#include <jni.h>
#include <ndkDTP.h> //JNI header
#include <TestClass.h> //C++ header
TestClass *m_globalTestClass;
void Java_com_test_ndktest_NDKTestActivity_initializeTestClass(JNIEnv* env, jobject javaThis) {
m_globalTestClass = new TestClass(env);
}
void Java_com_test_ndktest_NDKTestActivity_destroyTestClass(JNIEnv* env, jobject javaThis) {
delete m_globalTestClass;
m_globalTestClass = NULL;
}
jstring Java_com_test_ndktest_NDKTestActivity_invokeNativeFunction(JNIEnv* env, jobject javaThis) {
jstring testJS = m_globalTestClass->getString();
return testJS;
}
C++ intestazione:
class TestClass
{
public:
jstring m_testString;
JNIEnv *m_env;
TestClass(JNIEnv *env);
jstring getString();
};
C++ corpo:
#include <jni.h>
#include <string.h>
#include <TestClass.h>
TestClass::TestClass(JNIEnv *env){
m_env = env;
m_testString = m_env->NewStringUTF("TestClass: Test string!");
}
jstring TestClass::getString(){
return m_testString;
}
Grazie
Ecco fatto! Ero sicuro che il problema fosse altrove. Comunque, un grande aiuto. Molte grazie! – cierech