前面總結了靜態實現JNI的方法,本文介紹如何動態實現JNI:JNI在加載時,會調用JNI_OnLoad,而卸載時會調用JNI_UnLoad,因此咱們能夠經過在JNI_OnLoad裏面註冊咱們的native函數來實現JNI。下面就介紹該方法。html
1 Android應用層代碼java
在eclipse中新建工程NdkLoad,工程文件NdkLoad.java的代碼以下: android
package com.skywang.ndk; import android.app.Activity; import android.os.Bundle; import android.widget.TextView; import android.util.Log; public class NdkLoad extends Activity { public static final String TAG="skywang--NdkLoad"; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); Log.d(TAG, "on create"); TextView myTextView = new TextView(this); myTextView.setText( HelloLoad() ); setContentView(myTextView); } // jni中註冊的方法 public native String HelloLoad(); static { // 加載本地libndk_load.so庫文件 System.loadLibrary("ndk_load"); } }
public native String HelloLoad(); 這句話的做用是聲明HelloLoad()這個本地方法。HelloLoad()是經過jni中註冊到Android的方法,具體的實如今libndk_load.so中。
System.loadLibrary("ndk_load"); 這個函數的做用是加載libndk_load.so庫文件。因爲定義在NdkLoad類的static函數體中,因此在創建NdkLoad這個Acitivity時就會執行。數組
下面介紹ndk_load的具體實現。app
咱們知道,系統初始化JNI在加載時,會調用JNI_OnLoad(),而卸載時會調用JNI_UnLoad();因此,咱們能夠經過重寫JNI_OnLoad(),在JNI_OnLoad()中將函數註冊到Android中,以便能經過Java訪問。在本文中,咱們就是重寫JNI_OnLoad()函數實現ndk_load庫。eclipse
2 JNI動態註冊的實現方法ide
2.1 編寫JNI動態註冊的方法函數
(01) 打開終端,切換到NdkLoad所在目錄,新建jni目錄。學習
假設NdkLoad所在目錄爲"/home/skywang/workspace/android_apps/NdkLoad",則執行如下命令:ui
$ cd /home/skywang/workspace/android_apps/NdkLoad/ $ mkdir jni
(02) 在jni目錄下新建ndk_load.c,ndk_load.c的代碼以下:
#include <stdlib.h> #include <string.h> #include <stdio.h> #include <jni.h> #include <assert.h> // 獲取數組的大小 # define NELEM(x) ((int) (sizeof(x) / sizeof((x)[0]))) // 指定要註冊的類,對應完整的java類名 #define JNIREG_CLASS "com/skywang/ndk/NdkLoad" // 返回字符串"hello load jni" JNIEXPORT jstring JNICALL native_hello(JNIEnv *env, jclass clazz) { return (*env)->NewStringUTF(env, "hello load jni."); } // Java和JNI函數的綁定表 static JNINativeMethod method_table[] = { { "HelloLoad", "()Ljava/lang/String;", (void*)native_hello },//綁定 }; // 註冊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; }
JNI_OnLoad()會在JNI註冊時被調用。在JNI_OnLoad()中,調用register_ndk_load()。
register_ndk_load()調用registerNativeMethods()。
registerNativeMethods()中經過FindClass()找到class;而後經過RegisterNatives()將method_table註冊到class中。method_table是JNINativeMethod類型。
JNINativeMethod的定義以下:
typedef struct { const char* name; // Java中申明的Native函數名稱 const char* signature; // 描述了函數的參數和返回值 void* fnPtr; // 函數指針,指向C函數 } JNINativeMethod;
經過method_table,就將本地的native_hello()函數和註冊到Java中的HelloLoad()綁定起來了。當咱們在Java中調用HelloLoad()時,實際調用的是native_hello()。
(03) 在jni目錄下新建Android.mk,Android.mk的代碼以下:
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_MODULE := ndk_load LOCAL_SRC_FILES := ndk_load.c include $(BUILD_SHARED_LIBRARY) LOCAL_PATH := $(call my-dir)
3 編譯生成.so庫文件
切換到NdkLoad工程目錄,並執行ndk-build,生成.so庫文件。執行的命令以下:
$ cd /home/skywang/workspace/android_apps/NdkLoad/ $ ndk-build
命令執行成功,則生成「libs/armeabi/libndk_load.so」庫文件。若命令執行失敗,請先確保已經導入了ndk環境變量(請參考「Android JNI和NDK學習(01)--搭建NDK開發環境」)!
4 執行工程
如下是在平板上運行的實際效果圖:
點擊下載:源代碼