JNI技术2---本地代码反调用java类方法过程详解

记得在前面的JNI中我介绍了  在 java中调用本地代码的例子 ,这次 我就反过来   。 (如果初次接触JNI请看 JNI技术-1上篇文章)

首先我们在利用javah工具 生成 java类所对应的 C++头文件的时候 ,我们观察 java本地方法相关的定义 ..

JNIEXPORT void JNICALL Java_me_Native_Reflect_show (JNIEnv *, jobject); 

  //看这里2个参数 ,JNIEnv 类型的指针 ,和jobject类型的对象     他们分别代表了 JRE (java 运行时环境)  和 调用这个本地方法的 类的对象 或者 类的Class对象  。

JNIEnv其实是定义的一个结构体  里面包含了 许许多多的函数  比如 GetMethodID() 获取方法的 jmethodID对象     

                                                                                                                          GetStaticMethodID()获取静态方法的jmethodID对象 

                                                                                                                          CallStaticVoidMethod()调用静态 无返回值的方法等等  许许多多类似应用在JNI.H头文件中大家自己去查看   就清晰可见了  ,关于这2个对像是如何传递进来的呢 ?在调用本地方法的同时由JNI传递  比如我们担心

 

 因为 在JNI中 从 java到 C++有引用类型 和基本数据类型的过度,  也就是  在 java中没中数据类型 都有 一个与之对应的 C++类型 

  例如我们的 java中的 int 类型 在 C+中对于 jint类型 ,一般情况下都是在java数据类型 和引用类型前面加上一个 j开头 表示 是java类型的对应、

我们在 jni.h头文件中 找到如下定义

class _jobject {};
class _jclass : public _jobject {};
class _jthrowable : public _jobject {};
class _jstring : public _jobject {};
class _jarray : public _jobject {};
class _jbooleanArray : public _jarray {};
class _jbyteArray : public _jarray {};
class _jcharArray : public _jarray {};
class _jshortArray : public _jarray {};
class _jintArray : public _jarray {};
class _jlongArray : public _jarray {};
class _jfloatArray : public _jarray {};
class _jdoubleArray : public _jarray {};
class _jobjectArray : public _jarray {};
等等 的类型 还有很多 ,我们发现他们很有层次感  

所有对象 直接或者间接的继承了 _jobject类 ,这也是为了 和 java中 以object为基类的类层次相对应,

 

下面是C++和 java互相调用的2段代码 

在C++中饭调用java     在上次的操作之后修改 JNI方法的实现部分

#include <iostream>
#include "me_Native_Reflect.h"
using namespace std ;
JNIEXPORT void JNICALL Java_me_Native_Reflect_show(JNIEnv * env, jobject obj)//java本地方法的实现
{
     cout<<"java调用本地C++成功!"<<endl ;
  jclass cls =env->GetObjectClass(obj) ;  //获得JAVA对象在C++中的对应对象  jclass
  jmethodID  id=env->GetStaticMethodID(cls,"showStatic","()V") ;  //通过 JNIEnv类 也就是java环境获得  静态方法的 jmethodID          

//第三个参数 是方法的 返回类型参数签名  可以用  javap工具来查看 一个类的所有属性和方法  以及他们的签名  。。。。。
  env->CallStaticVoidMethod(cls,id) ;  //INVOKE  STATIC  METHOD   执行java静态方法 
}

 

//在java中调用C++   这是java代码

package me.Native;
public  class  Reflect
{
 public  native void  show()  ; 
 public  static void showStatic()
 {
  System.out.println("c++调用java静态方法成功!") ;
 }
 public  static  void main(String[]args)
 {  
  System.loadLibrary("JNIReflect") ;
  Reflect r=new Reflect() ;
  r.show() ;
 }

}

上一篇:Mysql学习之--Mysql二进制日志管理


下一篇:Spring boot打包war包