基於 Android NDK 的學習之旅----- C調用Java

來源:互聯網
上載者:User

標籤:des   android   c   class   blog   java   

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

基於 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、介面設計如下:

 

 

老樣子,很搓,不過實用,嘿嘿

代碼不在這貼出了,有需要的兄弟直接到文章結束部分下載。

 

 

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 <[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);
    }
 
}
  

 

 

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

 

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.