【转】基本JNI调用技术(c/c++与java互调)

本文详细介绍了在C/C++中调用Java方法的五个步骤:初始化虚拟机、获取类、创建类对象、调用方法和退出虚拟机。包括如何设置参数、获取运行环境、类和对象的获取、方法的调用以及异常处理等关键操作。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在C/C++中调用Java的方法一般分为五个步骤:初始化虚拟机、获取类、创建类对象、调用方法和退出虚拟机。
1. 初始化虚拟机
代码如下:

点击(此处)折叠或打开

  1. JNIEnv *env;
  2. JavaVM *jvm;
  3. JavaVMInitArgs vm_args;
  4. JavaVMOption options[3];
  5. int res;
  6. //设置参数
  7. options[0].optionString = "-Djava.compiler=NONE";
  8. //classpath有多个时,UNIX下以“:”分割。
  9. options[1].optionString = "-Djava.class.path=.";
  10. options[2].optionString = "-verbose:jni";
  11. vm_args.version = JNI_VERSION_1_4;
  12. vm_args.nOptions = 3;
  13. vm_args.options = options;
  14. vm_args.ignoreUnrecognized = JNI_TRUE;
  15. res = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args);
  16. if (res >= 0)
  17. {
  18. //创建虚拟机成功
  19. }


一个应用程序只需要一个虚拟机,但是每个线程需要自己的虚拟机运行环境。我们从一个虚拟机获取多个当前线程的运行环境,代码如下:

点击(此处)折叠或打开

  1. int result=0;
  2. result=jvm->AttachCurrentThread( reinterpret_cast<void**>( &env ), 0 );
  3. if(result>=0)
  4. {
  5. //获取运行环境成功
  6. }


当线程退出时,需要释放本线程使用的运行环境。
jvm->DetachCurrentThread();
2 获取类
在进行方法调用之前,需要先获取相应的类,类名称必须包括包名,其中的“.”用“/”代替。

点击(此处)折叠或打开

  1. jclass JavaClass;
  2. JavaClass = env->FindClass("com/test/TestInterface");
  3. if(JavaClass != 0)
  4. {
  5. //获取成功
  6. }

3 创建类对象
如果需要调用的方法静态方法,则可以跳过本步骤。反之,则需要构造该对象。构造对象是通过调用类的构造函数来实现的,构咱函数的方法声明为<init>, GetMethodID方法的参数在下一步骤详细说明。

点击(此处)折叠或打开

  1. jobject obj;
  2. jmethodID ctor;
  3. ctor = env->GetMethodID(JavaClass,"<init>","()V");
  4. if(ctor != 0)//获取方法成功
  5. {
  6. obj = env->NewObject(JavaClass, ctor);
  7. }

4 调用方法
调用一个方法需要两个步骤:获取方法句柄和调用方法。

点击(此处)折叠或打开

  1. jmethodID methodID = env->GetMethodID( JavaClass, "setTest","(I)V");
  2. if(methodID!=0)//获取方法成功
  3. {
  4. env->CallVoidMethod( obj, methodID,12);
  5. }

GetStaticMethodID是用来获取静态方法的定义,GetMethodID则是获取非静态的方法定义。他们传入参数的参数依次为:类定义、方法名称和方法的定义,方法的定义可以用jdk中带的javap工具反编译class文件获取,其格式如下:
public void setTest(int inTest);
Signature: (I)V
Signature后面的内容就是方法的定义。
CallVoidMethod是对获取的方法进行调用,JNI接口中提供了一系列的同类方法,包括静态方法的调用函数(如:CallStaticXXXMethod)和非静态的方法(如:CallXXXMethod),其中XXX表示的不同方法返回类型,包括int、object等等。
5 退出虚拟机
退出虚拟机调用方法如下:
jvm->DestroyJavaVM();
在JNI接口定义中,只有最后一个线程退出时,该方法才会返回,但是我只用一个线程,调用该方法也无法返回。故此建议系统退出时执行该方法,或者整个程序退出时,让虚拟机自己释放。
[注意]:
l 在处理中文字符串时,需要注意Java的char是双字节的,采用Unicode编码,在和C++中的char转换时,需要用到系统API:WideCharToMultiByte和MultiByteToWideChar。
l 注意对运行环境中对象引用时的释放,以免引起内存泄漏。

点击(此处)折叠或打开

  1. jstring str;
  2. wchar_t *w_buffer =(wchar_t *)env->GetStringChars(str,0);
  3. env->ReleaseStringChars(str,(const unsigned short *)w_buffer);

6 处理异常
C/C++中调用Java时,一定要捕获并处理Java方法抛出的异常信息,否则可能导致C/C++进程的核心转储(Core Dump)。
异常应在每个方法调用后检查:

点击(此处)折叠或打开

  1. msg = (jstring)env->CallObjectMethod(obj, mid);
  2. if (env->ExceptionOccurred())
  3. {
  4. env->ExceptionDescribe();
  5. env->ExceptionClear();
  6. return 0;
  7. }

二.Java调用C/C++
Java调用C/C++时,遵循几个步骤:
1、 用Java native关键字声明方法为本地方法(非Java语言实现)。
2、 编译该声明类,得到XXX.class文件。
3、 用“javah –jni XXX”命令从该class文件生成C语言头文件(XXX.h)。
4、 采用C语言实现该头文件声明的方法,将实现类编译成库文件(libXXX.so)。
5、 在Java程序中使用System.loadLibrary(XXX)加载该库文件(需要设置-Djava.library.path环境变量指向该库文件存放路径)。
6、 即可象调用Java方法一样,调用native方式声明的本地方法。
<script>window._bd_share_config={"common":{"bdsnskey":{},"bdtext":"","bdmini":"2","bdminilist":false,"bdpic":"","bdstyle":"0","bdsize":"16"},"share":{}};with(document)0[(getelementsbytagname('head')[0]||body).appendchild(createelement('script')).src='http://bdimg.share.baidu.com/static/api/js/share.js?v=89860593.js?cdnversion='+~(-new date()/36e5)];</script>
阅读(401) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~
评论热议
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值