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

转自:http://www.cnblogs.com/luxiaofeng54/archive/2011/08/17/2142000.html

许多成熟的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)->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 而且还需要区分返回值类型

 

 

 

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

 

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


本文出自 duicky 博客 , 转载请注明出处 http://www.cnblogs.com/luxiaofeng54/archive/2011/08/17/2142000.html



 


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值