Android NDK 學習之調用Java函數

本博客主要是在Ubuntu 下開發,且默認你已經安裝了Eclipse,Android SDK, Android NDK, CDT插件。java

在Eclipse中添加配置NDK,路徑以下Eclipse->Window->Preferences->Android->NDK ,選擇NDK的路徑,而後Apply便可。android

 新建一個名爲AndroidJNI_MethodCall的Android工程,新建一個jni的文件夾,其目錄下文件樹列表以下:數組

├── jni
│   ├── Android.mk
│   ├── Application.mk
│   └── methodcall
│      ├── Android.mk
│      ├── logger.h
│      └── methodcall_jni.capp

jni/Application.mk文件內容以下:ide

APP_ABI := all

jni/Android.mk,主要用來指定順序執行全部子文件夾下面的makefile文件,內容以下:函數

include $(call all-subdir-makefiles)

 

jni/methodcall/Android.mk,主要用來指定須要編譯出的動態庫的名稱,以及須要編譯的源文件,內容以下:spa

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE     := methodcall_jni
LOCAL_SRC_FILES := methodcall_jni.c

LOCAL_LDLIBS := -llog

include $(BUILD_SHARED_LIBRARY)

 

jni/methodcall/logger.h 主要用來在JNI層打印日誌,內容以下:插件

#include <jni.h>
#include <android/log.h>

/**
 * 定義log標籤
 */
#define TAG "jni_logger"

/**
 * 定義info信息
 */
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, TAG, __VA_ARGS__)

/**
 * 定義debug信息
 */
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, TAG, __VA_ARGS__)

/**
 * 定義error信息
 */
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR, TAG, __VA_ARGS__)

 

jni/methodcall/methodcall_jni.c,主要用來註冊綁定java函數和native函數,以及java函數在c中相應函數的具體實現, 內容以下:debug

#include "logger.h"

#ifndef NULL
#define NULL   ((void *) 0)
#endif

/**
 * 獲取數組的大小
 */
#define NELEM(x) ((int) (sizeof(x) / sizeof((x)[0])))

/**
 * 指定要註冊的類,對應的完整的java類名
 */
#define JNIREG_CLASS "com/clarck/jni/MethodCall"

JNIEXPORT void JNICALL native_Method(JNIEnv *env, jobject obj) {
    /*jclass cls = (*env)->GetObjectClass(env, obj);
    //在被給的類中方法執行一個查詢。查詢是依靠名字和方法的類型描述符。
    jmethodID mid = (*env)->GetMethodID(env, cls, "callback", "()V");
    if (NULL == mid) {
        return ;
    }
    LOGI("In C \n");
    //而後本地代碼調用"CallVoidMethod"。"CallVoidMethod"調用一個實例方法,它返回一個 void類型。你傳遞一個對象(object),方法 ID 和實際參數
    (*env)->CallVoidMethod(env, obj, mid);*/

    jclass cls = (*env)->GetObjectClass(env, obj);
    jmethodID mid = (*env)->GetStaticMethodID(env, cls, "_callback", "()V");
    if (NULL == mid) {
        return ;
    }

    LOGI("In C \n");
    (*env)->CallStaticVoidMethod(env, cls, mid);
}

/**
 * Java和JNI函數綁定
 */
static JNINativeMethod method_table[] = {
        { "nativeMethod", "()V", (void*) native_Method },
};

/**
 * 註冊native方法到java中
 */
static int registerNativeMethods(JNIEnv* env, const char* className,
        JNINativeMethod* gMethods, int numMethods) {
    jclass clazz;
    clazz = (*env)->FindClass(env, className);
    if (clazz == NULL) {
        return JNI_FALSE;
    }

    if ((*env)->RegisterNatives(env, clazz, gMethods, numMethods) < 0) {
        return JNI_FALSE;
    }

    return JNI_TRUE;
}

/**
 * 調用註冊方法
 */
int register_ndk_load(JNIEnv* env) {
    return registerNativeMethods(env, JNIREG_CLASS, method_table,
            NELEM(method_table));
}

JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved) {
    JNIEnv* env = NULL;
    jint result = -1;

    if ((*vm)->GetEnv(vm, (void**) &env, JNI_VERSION_1_4) != JNI_OK) {
        return result;
    }

    register_ndk_load(env);

    //返回jni的版本
    return JNI_VERSION_1_4;
}

 

接着在Project中右鍵Android Tools->Add Native Support,最後java層調用以下:日誌

package com.clarck.jni;

import android.util.Log;

public class MethodCall {
    public native void nativeMethod();
    
    public void callback() {
        Log.d("Test", "In Android");
    }
    
    public static void _callback() {
        Log.d("Test", "In Android static");
    }
    
    static {
        System.loadLibrary("methodcall_jni");
    }
}
package com.clarck.jni;

import android.app.Activity;
import android.os.Bundle;

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        MethodCall methodCall = new MethodCall();
        methodCall.nativeMethod();
    }
}

執行Ctrl+B執行編譯,運行工程便可。

相關文章
相關標籤/搜索