gpt4 book ai didi

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

转载 作者:可可西里 更新时间:2023-11-01 16:38:24 33 4
gpt4 key购买 nike

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

33 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com