继续浏览精彩内容
慕课网APP
程序员的梦工厂
打开
继续
感谢您的支持,我会继续努力的
赞赏金额会直接到老师账户
将二维码发送给自己后长按识别
微信支付
支付宝支付

基于 Android NDK 的学习之旅----- C调用Java

青春有我
关注TA
已关注
手记 1240
粉丝 205
获赞 1008

许多成熟的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、界面设计如下:


webp

老样子,很搓,不过实用,嘿嘿

代码不在这贴出了,有需要的兄弟直接到文章结束部分下载。

2、      关键代码说明

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

jclassTestProvider;

jobjectmTestProvider;

jmethodIDgetTime;

jmethodIDsayHello;

C 中映射 类

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

C中新建对象

   jmethodIDconstruction_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)/includeLOCAL_LDLIBS += -L$(SYSROOT)/usr/lib -llog

LOCAL_MODULE    := NDK_04
LOCAL_SRC_FILES := \
CToJava.c \
Provider.cinclude $(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方法打印出上层调用得到的时间,并且上层成功吐司出调用信息出来。


webp


webp


webp

2、点击 “C调用java非静态方法”按钮

C成功调用了sayHello 方法, 并成功接收到 C 传递的参数,和 吐司出相对应的信息


webp


webp



作者:Android高级架构探索
链接:https://www.jianshu.com/p/61ddb5e1e251


打开App,阅读手记
0人推荐
发表评论
随时随地看视频慕课网APP