Android JNI動態註冊Native 方法(實現IDA中改名)

1、Android應用層代碼:

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的具體實現。

我們知道,系統初始化JNI在加載時,會調用JNI_OnLoad(),而卸載時會調用JNI_UnLoad();所以,我們可以通過重寫JNI_OnLoad(),在JNI_OnLoad()中將函數註冊到Android中,以便能通過Java訪問。在本文中,我們就是重寫JNI_OnLoad()函數實現ndk_load庫。

2.JNI動態註冊的實現方法

(01) 打開終端,切換到NdkLoad所在目錄,新建jni目錄。

假設NdkLoad所在目錄爲"/home/skywang/workspace/android_apps/NdkLoad",則執行以下命令:

$ 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”庫文件。

4.測試效果



代碼下載鏈接:http://download.csdn.net/detail/hk9259/8410853


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章