许多成熟的 C 引擎要移植到 Android 平台上使用 , 一般都会 提供 一些接口 , Android sdk jdk 实现。
下文将会介绍 C 如何 通过 JNI 层调用 Java 的静态和非静态方法。
 

1、主要流程

1、  新建一个测试类 TestProvider.java
a)         该类提供了 2 个方法
b)        一个静态的方法 , 一个非静态的方法
2、  JNI 中新建 Provider.c
a)         该文件中需要把 Java 中的类 TestProvider 映射到 C
b)        TestProvider 的两个方法映射到 C
c)         新建 TestProvider 对象
d)        调用两个方法
3、  Android 上层 调用 JNI
4、  JNI 层调用 C
5、  C 层调用 Java 方法
 

2、设计实现

1 、界面设计如下:

老样子 , 很搓 , 不过实用 , 嘿嘿
代码不在这贴出了 , 有需要的兄弟直接到文章结束部分下载。
 
 
2、      关键代码说明
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);
 
注意 Get XXXMethodID  Call XXXMethod
第一个 XXX 表示的是映射方法的类型,如: 静态 跟非静态
第二个 XXX 表示 调用方法的返回值 ,如: Void,Object, 等等。(调用静态方法的时候 Call 后面要加 Static
 
详细 映射方法 调用方法 请参考 JNI 文档 ,这个很重要
 
3、      Java 上层 关键代码
TestProvider.Java 的两个方法
 
package com.duicky;
 
/**
 *
 *
 * @author luxiaofeng <454162034@qq.com>
 *
 */
public class TestProvider {
 
    public static 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());
    }
 
    public void 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);
    }
 
}
  
 
 
4、      Android.mk 文件 关键代码
LOCAL_PATH := $(call my-dir)
 
include $(CLEAR_VARS)
 
LOCAL_C_INCLUDES := $(LOCAL_PATH)/include
LOCAL_LDLIBS += -L$(SYSROOT)/usr/lib -llog
 
 
LOCAL_MODULE    := NDK_04
LOCAL_SRC_FILES := \
CToJava.c \
Provider.c
 
include $(BUILD_SHARED_LIBRARY)
  
老样子 , 不说了 , 你懂的。 如果不懂 , 嘎嘎 , 那就请点击 Android.mk 文件 简介
 
 
5、      JNI 文件夹下文件
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();
}
  

3、运行效果

1 、点击 C 调用 java 静态方法”按钮
 
C 成功调用了 Java 中的 getTime 方法 , 通过 C 方法打印出上层调用得到的时间 , 并且上层成功吐司出调用信息出来。

 


2
、点击 C 调用 java 非静态方法”按钮
 
C 成功调用了 sayHello 方法 , 并成功接收到 C 传递的参数,和 吐司出相对应的信息
 


 

4C调用Java注意点

       a) C 映射 java 方法时 对应的签名
getTime = (*jniEnv)->GetStaticMethodID(jniEnv, TestProvider, "getTime","()Ljava/lang/String;");
       故事情节还没发展这么快 , 下一章才会专门介绍下这个签名的使用
       b) 映射方法的时候需要区别静态和非静态 GetStaticMethodID,GetMethodID
    c) 调用的时候也需要区分 CallStaticObjectMethod,CallVoidMethod 而且还需要区分返回值类型
 
 
 
有不理解的兄弟请留言,个人技术有限,有讲错的地方请大牛们指出 , 讲的不够全面的请多多包涵,谢谢 ,
 
点击下载源码 C调用Java例子

本文出自 duicky 博客 , 转载请注明出处  http://www.cnblogs.com/luxiaofeng54/archive/2011/08/17/2142000.html