C++如何与Java进行交互?JNI在C++中的使用方法【跨语言编程】

3次阅读

JNI是java定义的接口规范,c++作为实现方被Java调用;C++也可回调Java方法,但需JNIEnv*和jvm环境。

C++如何与Java进行交互?JNI在C++中的使用方法【跨语言编程】

Java 和 C++ 交互主要靠 JNI(Java Native interface),它不是“C++ 调用 Java”,而是让 Java 虚拟机(JVM)能加载并调用 C++ 编写的本地函数——C++ 在这里扮演的是“被 Java 主动调用”的角色。反过来,C++ 也可以主动调用 Java 方法,但必须通过 JVM 环境(比如 AttachCurrentThread 或已存在的 JNIEnv*)。核心在于:JNI 是 Java 定义的接口规范,C++ 是实现方,不是对等通信协议。

1. 基础流程:从 Java 声明 native 到 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 命名规则实现函数(包名用下划线替换点,类名和方法名拼接):

#include <jni.h> JNIEXPORT jint JNICALL Java_Calculator_add(JNIEnv *env, jobject obj, jint a, jint b) {     return a + b; }

  • 函数名必须严格匹配:Java_全限定类名_方法名,其中点号(.)全部换成下划线(_)
  • 前两个参数固定:JNIEnv*(操作 JVM 的句柄)、jobject(当前 Java 对象实例,静态方法中为 jclass)
  • 返回值和参数类型要用 JNI 类型(jint、jstring、jobject 等),不能直接用 int 或 char*

2. 在 C++ 中调用 Java 方法(正向回调)

如果 C++ 需要反过来调用 Java 的某个方法(比如通知结果、触发回调),需持有有效的 JNIEnv* 和目标 Java 对象引用(jobject)。

立即学习Java免费学习笔记(深入)”;

C++如何与Java进行交互?JNI在C++中的使用方法【跨语言编程】

php中级教程之ajax技术

AJAX即“Asynchronous Javascript And XML”(异步JavaScript和XML),是指一种创建交互式网页应用的网页开发技术。它不是新的编程语言,而是一种使用现有标准的新方法,最大的优点是在不重新加载整个页面的情况下,可以与服务器交换数据并更新部分网页内容,不需要任何浏览器插件,但需要用户允许JavaScript在浏览器上执行。《php中级教程之ajax技术》带你快速

C++如何与Java进行交互?JNI在C++中的使用方法【跨语言编程】 2114

查看详情 C++如何与Java进行交互?JNI在C++中的使用方法【跨语言编程】

  • 在 Java 启动的线程中,JNIEnv* 可直接从 JNI 函数参数获得
  • 若在新创建的 C++ 线程中调用,必须先调用 AttachCurrentThread 获取 JNIEnv*,用完后调用 DetachCurrentThread
  • FindClass 获取 Java 类,GetMethodID 获取方法 ID,再用 CallXXXMethod(如 CallVoidMethod、CallObjectMethod)执行

jclass cls = env->FindClass("java/lang/String"); jmethodID ctor = env->GetMethodID(cls, "<init>", "(Ljava/lang/String;)V"); jstring arg = env->NewStringUTF("hello"); jobject strObj = env->NewObject(cls, ctor, arg);

3. 处理常见数据类型与内存管理

JNI 不自动管理跨语言对象生命周期,稍不注意就会内存泄漏或崩溃:

  • jstring → C 字符串:用 GetStringUTFChars() 获取,之后必须配对调用 ReleaseStringUTFChars();不要直接 reinterpret_cast
  • C 字符串 → jstring:用 NewStringUTF()(仅支持修改过的 UTF-8),或 NewString() + GetStringLength() 处理宽字符
  • 局部引用(Local Reference):如 NewObject、FindClass 返回的对象默认是局部引用,只在当前 JNI 调用内有效;跨多次 JNI 调用需转为全局引用(NewGlobalRef)
  • 数组操作:GetIntArrayElements() 获取指针,用完必须调用 ReleaseIntArrayElements(),否则可能引发 GC 异常或数据不一致

4. 编译与部署要点

不同平台编译方式略有差异,但关键原则一致:

  • Linux:用 g++ 编译为 .so,链接 libjvm.so(路径通常在 $JAVA_HOME/jre/lib/amd64/server/)
  • windows:用 MSVC 编译为 .dll,链接 jvm.lib(路径在 %JAVA_HOME%jrebinserverjvm.dll 对应的 import lib)
  • 头文件必须包含:jni.h(位于 $JAVA_HOME/include/ 及其子目录 include/linux 或 include/win32)
  • JVM 启动时可通过 -Djava.library.path=xxx 指定本地库搜索路径

基本上就这些。JNI 不复杂但容易忽略细节,尤其是线程绑定、引用管理和字符编码转换。实际项目中建议封装一层 C++ RaiI 工具类(如 JStringGuard、JLocalRef)来减少出错概率。

text=ZqhQzanResources