本博客主要是在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執行編譯,運行工程便可。