android - JNI 保持对对象的全局引用,使用其他 JNI 方法访问它。在多个 JNI 调用中保持 C++ 对象存活

标签 android c++ java-native-interface

我刚开始使用 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/

相关文章:

java - Localhost 解析与 android 模拟器的集成

android 应用没有注册 onTouch

c++ - 错误 : method was not declared in this scope (but it is included)

c++ - 为什么插槽作为字符串传递? Qt

c++ - 高效处理为redis(hiredis)的SET命令创建模板

java - '无法加载 JNI 共享库 "C:\Program Files\Java\jre7\bin\client\jvm.dll"'

c++ - 如何将共享库链接到另一个共享库

java - 使用compareTo增加应用程序版本名称

android - 错误 : ConnectionResult{statusCode=INTERNAL_ERROR, 分辨率 = null}

java - 什么会导致 Java native 函数(在 C 中)在进入时出现段错误?