博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
基于 Android NDK 的学习之旅----- C调用Java
阅读量:5046 次
发布时间:2019-06-12

本文共 6384 字,大约阅读时间需要 21 分钟。

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

 

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

 

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)

  

老样子,不说了,你懂的。 如果不懂,嘎嘎,那就请点击 简介

 

 

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 传递的参数,和 吐司出相对应的信息

 

 

 

 

 

 

4、C调用Java注意点

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

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

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

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

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

 

 

 

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

 

点击下载源码

 

本文出自 博客 , 转载请注明出处 

转载于:https://www.cnblogs.com/Free-Thinker/p/6148277.html

你可能感兴趣的文章
IIS的各种身份验证详细测试
查看>>
JavaScript特效源码(3、菜单特效)
查看>>
Linux常用命令总结
查看>>
yii模型ar中备忘
查看>>
C#线程入门
查看>>
CSS清除浮动方法
查看>>
JVM内存回收机制简述
查看>>
洛咕 P2480 [SDOI2010]古代猪文
查看>>
js-创建对象的几种方式
查看>>
JDK JRE Java虚拟机的关系
查看>>
2018.11.20
查看>>
word20161215
查看>>
12th week blog
查看>>
dijkstra (模板)
查看>>
python小记(3)
查看>>
编译Linux驱动程序 遇到的问题
查看>>
大型分布式网站架构技术总结
查看>>
HDU 1017[A Mathematical Curiosity]暴力,格式
查看>>
[算法之美] KMP算法的直观理解
查看>>
EntityFramework 性能优化
查看>>