我刚开始使用 JNI,但遇到以下问题。
我有一个包含简单类的 C++ 库。我从 Java Android 项目中调用了三个 JNI 方法,分别实例化所述类、调用实例化类的方法并销毁它。我保留了对该对象的全局引用,因此我可以在其他两个 JNI 方法中使用它。
我怀疑我做不到。当我运行该应用程序时,出现运行时错误(使用过时的引用),我怀疑这是因为全局引用在后续调用其他 JNI 方法时无效。
是实现我想要的(让对象在多个 JNI 调用中存在)的唯一方法,实际上将指向实例化类的指针传回 Java,将其保留在那里,然后将其传回 JNI功能?如果是这样,那很好,我想确保我不能用全局引用来做到这一点,而且我不仅仅是遗漏了什么。
我已经阅读了有关 JNI 中全局/局部引用的文档和章节,但它似乎只适用于 Java 类,而不适用于我自己的 native C++ 类,或者我错了。
如果我的描述不清楚,这里是代码(总结一下,我想知道这种持久化对象的机制是否会起作用):
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 header :
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 主体:
#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++ header :
class TestClass
{
public:
jstring m_testString;
JNIEnv *m_env;
TestClass(JNIEnv *env);
jstring getString();
};
C++ 主体:
#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;
}
谢谢
最佳答案
您的实现的问题是 jstring
数据成员。 NewStringUTF()
创建一个 Java String
对象以从 JNI 方法返回。所以它是一个 Java 本地引用。但是,您将其存储在 C++ 对象中并尝试在 JNI 调用中使用它。
您应该更好地区分 C++ 对象、Java 和它们之间的 JNI 接口(interface)。换句话说,C++ 应该使用 C++ 方式来存储字符串(如 std::string
)。 InvokeNativeFunction()
的 JNI 实现应将其转换为 jstring
作为返回值。
PS:有的情况需要 C++ 实现来保留对 Java 对象的引用(或相反)。但如果做得不好,它会使代码更加复杂并且容易出现内存错误。因此,您应该只在真正增加值(value)的地方使用它。
关于android - JNI 保持对对象的全局引用,使用其他 JNI 方法访问它。在多个 JNI 调用中保持 C++ 对象存活,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9796367/