JNI是Java定义的接口规范,C++作为实现方被Java调用;C++也可回调Java方法,但需JNIEnv*和JVM环境。
Java 和 C++ 交互主要靠 JNI(Java Native Interface),它不是“C++ 调用 Java”,而是让 Java 虚拟机(JVM)能加载并调用 C++ 编写的本地函数——C++ 在这里扮演的是“被 Java 主动调用”的角色。反过来,C++ 也可以主动调用 Java 方法,但必须通过 JVM 环境(比如 AttachCurrentThread 或已存在的 JNIEnv*)。核心在于:JNI 是 Java 定义的接口规范,C++ 是实现方,不是对等通信协议。
Java 端先声明一个 native 方法,并加载对应动态库:
public class Calculator {
static {
System.loadLibrary("calculator"); // 加载 libcalculator.so(Linux)或 calculator.dll(Windows)
}
public native int add(int a, int b);
}
C++ 端需按 JNI 命名规则实现函数(包名用下划线替换点,类名和方法名拼接):
#includeJNIEXPORT jint JNICALL Java_Calculator_add(JNIEnv *env, jobject obj, jint a, jint b) { return a + b; }
如果 C++ 需要反过来调用 Java 的某个方法(比如通知结果、触发回调),需持有有效的 JNIEnv* 和目标 Java 对象引用(jobject)。
jclass cls = env->FindClass("java/lang/String"); jmethodID ctor = env->GetMethodID(cls, "
", "(Ljava/lang/String;)V"); jstring arg = env->NewStringUTF("hello"); jobject strObj = env->NewObject(cls, ctor, arg);
JNI 不自动管理跨语言对象生命周期,稍不注意就会内存泄漏或崩溃:
不同平台编译方式略有差异,但关键原则一致:
基本上就这些。JNI 不复杂但容易忽略细节,尤其是线程绑定、引用管理和字符编码转换。实际项目中建议封装一层 C++ RAII 工具类(如 JStringGuard、JLocalRef)来减少出错概率。