Android JNI實例

Android的SDK中沒有包括JNI的支持,並且對如何支持JNI也沒有任何文檔說明。不過既然整個Android平臺是開源的,咱們能夠經過Google發佈的源代碼來找到一些線索(好比frameworks/base/media/jni/目錄),依葫蘆畫瓢的實現上層JAVA程序經過JNI來調用Native C程序中的函數。 html

依照下面的步驟能夠實現一個很是簡單的JNI的實例程序: java

1.  首先編寫C模塊,實現動態庫。(關於如何在Android中編譯C模塊的更多細節,請參考《Android編譯環境(1) - 編譯Native C的helloworld模塊》。) android

development目錄下添加新目錄hellolib,並添加hellolib.cAndroid.mk文件。hellolib.c的內容以下: web

#include <jni.h> app

#define LOG_TAG "TestLib" ionic

#undef LOG ide

#include <utils/Log.h> 函數

 

JNIEXPORT void JNICALL Java_com_test_TestHelloLib_printHello(JNIEnv * env, jobject jobj) 工具

{ 測試

    LOGD("Hello LIB!\n");

}

注意這裏的函數名須要按照JNI的規範(所以也能夠用javah -jni工具來生成頭文件,來保證函數名的正確性),Java_com_test_TestHelloLib_printHello的命名對應後面在java代碼中,package名字是com.test,類名是TestHelloLibnative函數名是printHello

另外,LOGD#define LOG_TAG "TestLib"等打印log的方式是採用了Android所提供的LOG機制,這樣才能經過Androidlogcat工具看到log

用於編譯C模塊的Android.mk文件內容以下:

LOCAL_PATH:= $(call my-dir)

include $(CLEAR_VARS)

LOCAL_SRC_FILES:= \

    hellolib.c

LOCAL_C_INCLUDES := \

    $(JNI_H_INCLUDE)

 

LOCAL_SHARED_LIBRARIES := \

    libutils

LOCAL_PRELINK_MODULE := false

LOCAL_MODULE := libhello

include $(BUILD_SHARED_LIBRARY)

該文件中的一些變量分別對應的含義以下:

LOCAL_SRC_FILES - 編譯的源文件

LOCAL_C_INCLUDES - 須要包含的頭文件目錄

LOCAL_SHARED_LIBRARIES - 連接時須要的外部庫

LOCAL_PRELINK_MODULE - 是否須要prelink處理(參考prelink的詳細介紹:《動態庫優化——Prelink(預鏈接)技術》,AndroidToolchain, prelink工具:《Android Toolchain與Bionic Libc》)

LOCAL_MODULE - 編譯的目標對象

BUILD_SHARED_LIBRARY - 指明要編譯成動態庫。

       接下來回到Android頂層目錄,並執行make libhello來編譯:

# cd $(YOUR_ANDROID) && make libhello

target thumb C: libhello <= development/hellolib/hellolib.c

target SharedLib: libhello (out/target/product/generic/obj/SHARED_LIBRARIES/libhello_intermediates/LINKED/libhello.so)

target Non-prelinked: libhello (out/target/product/generic/symbols/system/lib/libhello.so)

target Strip: libhello (out/target/product/generic/obj/lib/libhello.so)

Install: out/target/product/generic/system/lib/libhello.so

       編譯結果可獲得位於out/target/product/generic/system/lib/目錄的動態共享庫libhello.so

 

2.編寫Java模塊,來經過JNI方式調用C接口。具體Eclipse環境的搭建請參考Android SDK文檔中的詳細說明,及Hello Android程序的建立過程,這裏僅給出咱們須要修改的TestHelloLib.java文件:

package com.test;

import android.app.Activity;

import android.os.Bundle;

public class TestHelloLib extends Activity {

    /** Called when the activity is first created. */

    @Override

    public void onCreate(Bundle savedInstanceState) {

        super.onCreate(savedInstanceState);

        setContentView(R.layout.main);

        printHello();

    }

    static {

    System.loadLibrary("hello");

    }

    private native void printHello();

}

注意上面代碼中粗體字部分:private native void printHello()用來聲明一個native接口,static { System.loadLibrary("hello"); } 用來加載上面步驟中生成libhello.so(注意loadLibrary方法的參數不是」libhello.so」,而是去掉前綴和後綴以後的」hello」),onCreate()方法中則調用了printHello()接口。

    經過這一步驟可生成Android開發者所熟悉的apk文件:TestHelloLib.apk。

 

3.集成測試TestHelloLib.apk和libhello.so。先運行emulator並將TestHelloLib.apk和libhello.so上傳至emulator中。注意要將libhello.so上傳到emulator/system/lib目錄,因爲該目錄是隻讀的,上傳以前先要執行adb remount

# adb remount

# adb push out/target/product/generic/system/lib/libhello.so /system/lib

# adb install TestHelloLib.apk

       接下來在模擬器菜單中能夠看到已經安裝的TestHelloLib程序,運行便可。

       因爲JNI接口printHello()並無做界面上的改動,要驗證其效果須要用Androidlogcat工具來查看。運行」adb logcat」能夠找到下面的log片段:

I/ActivityManager(   48): Starting activity: Intent { action=android.intent.action.MAIN categories={android.intent.category.LAUNCHER} flags=0x10200000 comp={com.test/com.test.TestHelloLib} }

I/ActivityManager(   48): Start proc com.test for activity com.test/.TestHelloLib: pid=174 uid=10024 gids={}

D/dalvikvm(  174): Trying to load lib /system/lib/libhello.so 0x43481c58

D/dalvikvm(  174): Added shared lib /system/lib/libhello.so 0x43481c58

D/dalvikvm(  174): No JNI_OnLoad found in /system/lib/libhello.so 0x43481c58

D/dalvikvm(  174): +++ not scanning '/system/lib/libwebcore.so' for 'printHello' (wrong CL)

D/dalvikvm(  174): +++ not scanning '/system/lib/libmedia_jni.so' for 'printHello' (wrong CL)

D/TestLib (  174): Hello LIB!

I/ActivityManager(   48): Displayed activity com.test/.TestHelloLib: 806 ms

       這裏包含了調用printHello()接口的log信息,其中D/TestLib (  174): Hello LIB!」就是printHello()所打印的信息。至此成功完成Android JNI的實例驗證。

相關文章
相關標籤/搜索