C++如何通过JNI调用Java_C++与Java混合编程及JNI接口使用

答案:本文介绍C++通过JNI调用Java方法的实现方式,依次讲解获取JNIEnv和JavaVM的方法、调用Java对象与静态成员的步骤、以及异常处理和资源管理。1. 通过JNI_OnLoad保存JavaVM,非Java线程需AttachCurrentThread获取JNIEnv;2. 调用实例方法需获取jclass、GetMethodID并使用CallXXXMethod;3. 调用静态方法使用GetStaticMethodID和CallStaticXXXMethod;4. 必须检查Exception并清理局部引用,避免内存泄漏。核心是掌握引用生命周期与线程安全。

在跨平台开发中,C++与Java混合编程是一种常见需求,尤其是在Android底层开发或性能敏感模块中。通过JNI(Java Native Interface),Java可以调用C++代码,反之亦然。本文重点讲解如何在C++中通过JNI调用Java方法,实现双向交互。

1. 获取JNIEnv和JavaVM

要从C++调用Java方法,必须获取有效的JNIEnv*指针,它是调用大多数JNI函数的基础。如果C++代码运行在非Java线程中,还需先通过JavaVM附加线程。

说明: - JavaVM* 是每个JVM进程只有一个的全局对象,可用于创建或附加线程。 - JNIEnv* 是线程局部的,每个线程需独立获取。

示例:在C++中保存并使用JavaVM

// 在JNI_OnLoad中保存JavaVM

JavaVM* g_jvm = nullptr;
jint JNI_OnLoad(JavaVM* vm, void* reserved) {
    g_jvm = vm;
    return JNI_VERSION_1_6;
}

后续在线程中获取JNIEnv:

JNIEnv* env = nullptr;
if (g_jvm->GetEnv((void**)&env, JNI_VERSION_1_6) != JNI_OK) {
    g_jvm->AttachCurrentThread(&env, nullptr);
}

2. 调用Java对象的方法

从C++调用Java方法需以下步骤:

  • 获取对应的 jclass
  • 查找目标方法的 jmethodID
  • 使用 CallXXXMethod 系列函数调用

示例:调用Java对象的printMessage方法

jclass clazz = env->FindClass("com/example/MyActivity");
jmethodID methodId = env->GetMethodID(clazz, "printMessage", "(Ljava/lang/String;)V");
jstring msg = env->NewStringUTF("Hello from C++!");
jobject obj = /* 假设已有Java对象引用 */;
env->CallVoidMethod(obj, methodId, msg);

// 注意:字符串、对象使用后可根据需要DeleteLocalRef

env->DeleteLocalRef(msg);
env->DeleteLocalRef(clazz);

3. 访问Java静态方法和字段

调用静态方法与实例方法类似,但使用GetStaticMethodID和CallStaticXXXMethod。

示例:调用Java静态方法

jclass clazz = env->FindClass("com/example/Utils");
jmethodID mid = env->GetStaticMethodID(clazz, "log", "(Ljava/lang/String;)V");
jstring str = env->NewStringUTF("From C++ static");
env->CallStaticVoidMethod(clazz, mid, str);
env->DeleteLocalRef(str);
env->DeleteLocalRef(clazz);

读取或设置静态字段也类似,使用GetStaticFieldID和GetStaticXXXField / SetStaticXXXField。

4. 处理异常和资源管理

JNI调用可能抛出Java异常,C++中应检查并处理,避免崩溃。

示例:检查并清除异常

env->CallVoidMethod(obj, methodId);
if (env->ExceptionCheck()) {
    env->ExceptionDescribe(); // 打印异常栈
    env->ExceptionClear(); // 清除异常状态
}

建议:

  • 每次FindClass返回的是局部引用,使用后应DeleteLocalRef
  • 长期使用的类或方法ID可缓存为全局引用
  • 多线程中注意JNIEnv不能跨线程使用

基本上就这些。掌握JNIEnv获取、方法查找与调用、异常处理,就能在C++中灵活调用Java代码。关键是理解JNI的引用机制和生命周期管理,避免内存泄漏或崩溃。不复杂但容易忽略细节。