虽不是很详细,但步骤都已相当明确了的.


 一. C/C++


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


1.  初始化虚拟机


代码如下:

JNIEnv *env; 

 

       JavaVM *jvm; 

 

      JavaVMInitArgs vm_args; 

 

      JavaVMOption options[3]; 

 

      int res; 

 

      //设置参数 

 

  options[0].optionString = "-Djava.compiler=NONE"; 

 

  //classpath有多个时,UNIX下以“:”分割。 

 

      options[1].optionString = "-Djava.class.path=."; 

 

      options[2].optionString = "-verbose:jni"; 

 

    

 

      vm_args.version = JNI_VERSION_1_4; 

 

      vm_args.nOptions = 3; 

 

      vm_args.options = options; 

 

      vm_args.ignoreUnrecognized = JNI_TRUE; 

 

      res = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args); 

 

       if (res >= 0) 

 

  { 

 

       //创建虚拟机成功 

 

  }


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


int result=0; 

 

  result=jvm->AttachCurrentThread( reinterpret_cast<void**>( &env ), 0 ); 

 

  if(result>=0) 

 

  { 

 

       //获取运行环境成功 

 

  }


当线程退出时,需要释放本线程使用的运行环境。


jvm->DetachCurrentThread();


2 获取类


在进行方法调用之前,需要先获取相应的类,类名称必须包括包名,其中的“.”用“/”代替。

jclass JavaClass; 

 

  JavaClass = env->FindClass("com/test/TestInterface"); 

 

     if(JavaClass != 0) 

 

     { 

 

              //获取成功 

 

     }

3 创建类对象


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

jobject obj; 

 

  jmethodID ctor; 

 

  ctor = env->GetMethodID(JavaClass,"<init>","()V"); 

 

  if(ctor != 0)//获取方法成功 

 

     { 

 

           obj = env->NewObject(JavaClass, ctor); 

 

     }

4 调用方法


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


jmethodID methodID = env->GetMethodID( JavaClass, "setTest","(I)V"); 

 

  if(methodID!=0)//获取方法成功 

 

  { 

 

  env->CallVoidMethod( obj, methodID,12); 

 

  }


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  注意对运行环境中对象引用时的释放,以免引起内存泄漏。


jstring str; 

 

  wchar_t *w_buffer =(wchar_t *)env->GetStringChars(str,0); 

 

  env->ReleaseStringChars(str,(const unsigned short *)w_buffer);


6 处理异常


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


异常应在每个方法调用后检查:


msg = (jstring)env->CallObjectMethod(obj, mid); 

 
 
         if (env->ExceptionOccurred()) 

 
 
         { 

 
 
            env->ExceptionDescribe();           

 
 
            env->ExceptionClear(); 

 
 
            return 0; 

 
 
         }


二.Java


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方式声明的本地方法。