c 能调用java语言吗,C调用Java

基于 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、界面设计如下:

0818b9ca8b590ca3270a3433284dd417.png

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

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

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,"", "()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

#include

void

GetTime() ;

void

SayHello();

Provider.c

#include "Provider.h"

#include

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,

""

,

"()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

#include

#include

#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方法打印出上层调用得到的时间,并且上层成功吐司出调用信息出来。

0818b9ca8b590ca3270a3433284dd417.png

0818b9ca8b590ca3270a3433284dd417.png

0818b9ca8b590ca3270a3433284dd417.png

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

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

0818b9ca8b590ca3270a3433284dd417.png

0818b9ca8b590ca3270a3433284dd417.png

4、C调用Java注意点

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

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

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

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

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

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值