1、背景
在Report_Center_Task_Proc上报消息线程中,我们在消息队列里面获取到数据后,把数据简单解析后通过java接口g_jniEnv->CallVoidMethod上报数据,那什么在c++代码中怎么调用java函数方法的呢?
首先,JNI接口初始化会传入JNIEnv *env, jobject instance。
JNIEXPORT void JNICALL Java_com_starcor_data_needle_crawler_crawl_NeedleJNI_init(JNIEnv *env, jobject instance)
2、JNIEnv对象
JNIEnv类型代表Java环境。通过这个JNIEnv*指针,就可以对Java端的代码进行操作。如,创建Java类的对象,调用Java对象的方法,获取Java对象的属性等。
JNIEnv的指针会被JNI传送到本地方法的实现函数中来对Java端的代码进行操作
JNIEnv类中的函数:
NewObject/NewString/New<TYPE>Array :new新对象
Get/Set<TYPE>Field:获取属性
Get/SetStatic<TYPE>Field :获取静态属性
Call<TYPE>Method/CallStatic<TYPE>Method:调用方法
我们的接口调用
init rp->updateContext(JNIEnv *_jniEnv, jobject _jobject);
_jniEnv->GetJavaVM(&gJVM);
g_jobject = _jniEnv->NewGlobalRef(_jobject);
使用时通过gJVM获取到g_jniEnv:
gJVM->GetEnv((void **)&g_jniEnv, JNI_VERSION_1_6);
3、获取jclass
为了能够在C/C++使用Java类,jni.h头文件中专门定义了jclass类型来表示Java中的Class类
jclass的取得:
JNIEnv类中有如下几个简单的函数可以取得jclass
jclass FindClass(const char* clsName) 根据类名来查找一个类,完整类名。
jclass GetObjectClass(jobject obj) 根据一个对象,获取该对象的类
jclass GetSuperClass(jclass obj) 获取一个类的父类
FindClass 会在classpath系统环境变量下寻找类,需要传入完整的类名,注意包与包之间是用"/"而不是"."来分割
如:jclass cls_string= env->FindClass("java/lang/String");
获取jclass又什么用,比如你要调用类的静态方法,静态属性就需要通过这个方法来获取一个类。
我们的接口调用:
jclass eventClass = g_jniEnv->GetObjectClass(g_jobject);
4、本地代码访问Java类中的属性与方法
有了类和对象之后,如何才能访问java中的对象的属性和方法呢,这就需要用到以下这些方法了。
JNI在jni.h头文件中定义了jfieldID,jmethodID类表示Java端的属性和方法
如何获取属性: 在访问或设置Java属性的时候,首先就要现在本地代码中取得代表Java属性的jfieldID,然后才能在本地代码中进行Java属性操作。
如何调用java的方法:调用Java端的方法时,需要取得代表方法的jmethodID才能进行Java方法调用
JNIEnv获取相应的fieldID和jmethodID的方法:
GetFieldID/GetMethodID
GetStaticFieldID/GetStaticMethodID
GetMethodID也可以取得构造函数的jmethodID。创建Java对象时调用指定的构造函数。
如:env->GetMethodID(data_Clazz,"method_name","()V")
(*jniEnv)->GetMethodID(jniEnv, Clazz,"<init>", "()V");
这个比较特殊,这个是默认构造函数的方法,一般用这个来初始化对象,但是再实际过程中,为了快速生成一个实例,一般通过工厂方法类创建jobject
jni.h 对GetMethodID的定义:
jmethodID (JNICALL *GetMethodID)
(JNIEnv *env, jclass clazz, const char *name, const char *sig);
我们的接口调用:
jmethodID event_cb = g_jniEnv->GetMethodID(eventClass, "reportHttpPackageData", DEF_METH);
DEF_METH的定义:
#define DEF_METH "(Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;[BLjava/lang/String;[BLjava/lang/String;I)V"
5、数据转换
jstring flowkey = char2Jsting(g_jniEnv, MsgData->flowKey);
6、调用java方法event_cb上报数据
g_jniEnv->CallVoidMethod(g_jobject, \
event_cb, \
flowkey, \
flowtime, \
filterid, \
requsthead, \
requst, \
responsehead, \
response, \
statistics, \
MsgData->State);
就是调用java里面NeedleJNI 的reportHttpPackageData方法。