NDK 之 C调用Java

来源:互联网 发布:浙江省大数据科技协会 编辑:程序博客网 时间:2024/05/18 02:04
1、 新建一个测试类TestProvider.java

该类提供了2个方法

一个静态的方法,一个非静态的方法

2、JNI中新建Provider.c

该文件中需要把Java中的类TestProvider映射到C中

把TestProvider的两个方法映射到C中

新建TestProvider 对象

调用两个方法

3、 Android 上层 调用 JNI层

4、 JNI层调用C层

5、C 层调用 Java 方法

 

C中定义映射的类、方法、对象

jclass TestProvider;

jobject mTestProvider;

jmethodID getTime;

jmethodID sayHello;

C 中映射 类

       TestProvider = (*jniEnv)->FindClass(jniEnv,"com/duicky/TestProvider");

C中新建对象

       jmethodID construction_id = (*jniEnv)->GetMethodID(jniEnv, TestProvider,"<init>", "()V");

      TestProvider mTestProvider = (*jniEnv)->NewObject(jniEnv, TestProvider,construction_id);

C 中映射方法

       静态:getTime = (*jniEnv)->GetStaticMethodID(jniEnv, TestProvider, "getTime","()Ljava/lang/String;");

       非静态:sayHello = (*jniEnv)->GetMethodID(jniEnv, TestProvider, "sayHello","(Ljava/lang/String;)V");

C 中调用 Java的 方法

       静态:(*jniEnv)->CallStaticObjectMethod(jniEnv, TestProvider, getTime);

       非静态:(*jniEnv)->CallVoidMethod(jniEnv, mTestProvider, sayHello,jstrMSG);

注意 GetXXXMethodID  和  CallXXXMethod 。

第一个XXX 表示的是映射方法的类型,如: 静态跟非静态

第二个 XXX 表示 调用方法的返回值 ,如:Void,Object,等等。(调用静态方法的时候Call后面要加Static)

详细 映射方法和 调用方法 请参考 JNI 文档 ,这个很重要 !

 

public class TestProvider {
    publicstatic String getTime() {
        LogUtils.printWithSystemOut( "Call From C Java Static Method"  );
        LogUtils.toastMessage(MainActivity.mContext, "Call From C Java Static Method"  );
        return String.valueOf(System.currentTimeMillis());
    }
    publicvoid sayHello(String msg) {
        LogUtils.printWithSystemOut("Call From C Java Not Static Method :"+ msg);
        LogUtils.toastMessage(MainActivity.mContext, "Call From C Java Not Static Method :"+ msg);
    }
}

 

 

 

Provider.h

?
#include <string.h>
#include <jni.h>
void GetTime() ;
void SayHello();

Provider.c 

?
#include "Provider.h"
#include <android/log.h>
extern JNIEnv* jniEnv;
jclass TestProvider;
jobject mTestProvider;
jmethodID getTime;
jmethodID sayHello;
int GetProviderInstance(jclass obj_class);
/**
* 初始化 类、对象、方法
*/
int InitProvider() {
__android_log_print(ANDROID_LOG_INFO, "JNIMsg","InitProvider Begin  1" );
if(jniEnv == NULL) {
return 0;
}
if(TestProvider == NULL) {
TestProvider = (*jniEnv)->FindClass(jniEnv,"com/duicky/TestProvider");
if(TestProvider == NULL){
return -1;
}
__android_log_print(ANDROID_LOG_INFO, "JNIMsg","InitProvider Begin  2 ok" );
}
if (mTestProvider == NULL) {
if (GetProviderInstance(TestProvider) != 1) {
(*jniEnv)->DeleteLocalRef(jniEnv, TestProvider);
return -1;
}
__android_log_print(ANDROID_LOG_INFO, "JNIMsg","InitProvider Begin  3 ok" );
}
if (getTime == NULL) {
getTime = (*jniEnv)->GetStaticMethodID(jniEnv, TestProvider, "getTime","()Ljava/lang/String;");
if (getTime == NULL) {
(*jniEnv)->DeleteLocalRef(jniEnv, TestProvider);
(*jniEnv)->DeleteLocalRef(jniEnv, mTestProvider);
return -2;
}
__android_log_print(ANDROID_LOG_INFO, "JNIMsg","InitProvider Begin  4 ok" );
}
if (sayHello == NULL) {
sayHello = (*jniEnv)->GetMethodID(jniEnv, TestProvider, "sayHello","(Ljava/lang/String;)V");
if (sayHello == NULL) {
(*jniEnv)->DeleteLocalRef(jniEnv, TestProvider);
(*jniEnv)->DeleteLocalRef(jniEnv, mTestProvider);
(*jniEnv)->DeleteLocalRef(jniEnv, getTime);
return -3;
}
__android_log_print(ANDROID_LOG_INFO, "JNIMsg","InitProvider Begin  5 ok" );
}
__android_log_print(ANDROID_LOG_INFO, "JNIMsg","InitProvider Begin  6" );
return 1;
}
int GetProviderInstance(jclass obj_class) {
if(obj_class == NULL) {
return 0;
}
jmethodID construction_id = (*jniEnv)->GetMethodID(jniEnv, obj_class,
"<init>", "()V");
if (construction_id == 0) {
return -1;
}
mTestProvider = (*jniEnv)->NewObject(jniEnv, obj_class,
construction_id);
if (mTestProvider == NULL) {
return -2;
}
return 1;
}
/**
* 获取时间 ---- 调用 Java 方法
*/
void GetTime() {
if(TestProvider == NULL || getTime == NULL) {
int result = InitProvider();
if (result != 1) {
return;
}
}
jstring jstr = NULL;
char* cstr = NULL;
__android_log_print(ANDROID_LOG_INFO, "JNIMsg","GetTime Begin" );
jstr = (*jniEnv)->CallStaticObjectMethod(jniEnv, TestProvider, getTime);
cstr = (char*) (*jniEnv)->GetStringUTFChars(jniEnv,jstr, 0);
__android_log_print(ANDROID_LOG_INFO, "JNIMsg","Success Get Time from Java , Value = %s",cstr );
__android_log_print(ANDROID_LOG_INFO, "JNIMsg","GetTime End" );
(*jniEnv)->ReleaseStringUTFChars(jniEnv, jstr, cstr);
(*jniEnv)->DeleteLocalRef(jniEnv, jstr);
}
/**
* SayHello ---- 调用 Java 方法
*/
void SayHello() {
if(TestProvider == NULL || mTestProvider == NULL || sayHello == NULL) {
int result = InitProvider() ;
if(result != 1) {
return;
}
}
jstring jstrMSG = NULL;
jstrMSG =(*jniEnv)->NewStringUTF(jniEnv, "Hi,I'm From C");
__android_log_print(ANDROID_LOG_INFO, "JNIMsg","SayHello Begin" );
(*jniEnv)->CallVoidMethod(jniEnv, mTestProvider, sayHello,jstrMSG);
__android_log_print(ANDROID_LOG_INFO, "JNIMsg","SayHello End" );
(*jniEnv)->DeleteLocalRef(jniEnv, jstrMSG);
}

       CToJava.c

?
#include <string.h>
#include <android/log.h>
#include <jni.h>
#include "Provider.h"
JNIEnv* jniEnv;
/**
*  Java 中 声明的native getTime 方法的实现
*/
void Java_com_duicky_MainActivity_getTime(JNIEnv* env, jobject thiz)
{
if(jniEnv == NULL) {
jniEnv = env;
}
GetTime();
}
/**
*  Java 中 声明的native sayHello 方法的实现
*/
void Java_com_duicky_MainActivity_sayHello(JNIEnv* env, jobject thiz)
{
if (jniEnv == NULL) {
jniEnv = env;
}
SayHello();
}

C调用Java注意点

a) C 映射java 方法时 对应的签名

     getTime = (*jniEnv)->GetStaticMethodID(jniEnv, TestProvider, "getTime","()Ljava/lang/String;");

b)映射方法的时候需要区别静态和非静态GetStaticMethodID,GetMethodID

c)调用的时候也需要区分CallStaticObjectMethod,CallVoidMethod 而且还需要区分返回值类型

 

 

原创粉丝点击