基于 Android(安卓)NDK 的学习之旅----- C调用Java
16lz
2021-01-25
许多成熟的 C 引擎要移植到 Android 平台上使用 , 一般都会 提供 一些接口 , 让 Android sdk 和 jdk 实现。 下文将会介绍 C 如何 通过 JNI 层调用 Java 的静态和非静态方法。
老样子 , 很搓 , 不过实用 , 嘿嘿 代码不在这贴出了 , 有需要的兄弟直接到文章结束部分下载。 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 的两个方法
4、 Android.mk 文件 关键代码
老样子 , 不说了 , 你懂的。 如果不懂 , 嘎嘎 , 那就请点击 Android.mk 文件 简介 5、 JNI 文件夹下文件 Provider.h
Provider.c
CToJava.c
2 、点击 “ C 调用 java 非静态方法”按钮 C 成功调用了 sayHello 方法 , 并成功接收到 C 传递的参数,和 吐司出相对应的信息
本文出自 duicky 博客 , 转载请注明出处 http://www.cnblogs.com/luxiaofeng54/archive/2011/08/17/2142000.html
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 <[email protected]> |
* |
*/ |
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); |
} |
} |
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) |
#include <string.h> |
#include <jni.h> |
void GetTime() ; |
void SayHello(); |
#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); |
} |
#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例子本文出自 duicky 博客 , 转载请注明出处 http://www.cnblogs.com/luxiaofeng54/archive/2011/08/17/2142000.html
更多相关文章
- Nginx系列教程(四)| 一文带你读懂Nginx的动静分离
- Android设备的内置存储和外置存储到底是怎么回事,深入理解
- [置顶] 关于Socket的简单解析
- Android开发 如何最优的在Activity里释放资源
- Android(安卓)6.0 新特性
- Android(安卓)IPC机制(三)在Android(安卓)Studio中使用AIDL实现跨
- Android代码优化
- PHP简单判断iPhone、iPad、Android及PC设备的方法
- 自定义View关于measure流程的基本思路整理