«

C调用Java

时间:2024-3-2 17:07     作者:韩俊     分类: Java


本文出自 duicky 博客 , 转载请注明出处 http://www.cnblogs.com/luxiaofeng54/archive/2011/08/17/2142000.html
基于 Android NDK 的学习之旅----- C调用Java

许多成熟的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);

注意 GetXXXMethodID 和 CallXXXMethod 。

第一个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)-&gt;GetMethodID(jniEnv, obj_class,
        &quot;&lt;init&gt;&quot;,

"()V");

if (construction_id == 0) {
    return -1;
}

mTestProvider = (*jniEnv)-&gt;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 传递的参数,和 吐司出相对应的信息



4、C调用Java注意点

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

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

故事情节还没发展这么快,下一章才会专门介绍下这个签名的使用

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

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

有不理解的兄弟请留言,个人技术有限,有讲错的地方请大牛们指出,讲的不够全面的请多多包涵,谢谢,

点击下载源码 C调用Java例子



标签: java

热门推荐