android studio很好的一個JNI 入門博文
最近在將一個小應用從eclipse開發遷移到android studio,程式中有native程式碼實現,在eclipse是靠Android.mk這麼個mk檔案來組織編譯的,但到android studio上就不行了,因其是靠gradle組織,所以makefile裡的配置要轉換為build.gradle裡的語句(儘管實際上gradle也是組織一個mk檔案),同時在遷移過程中遇到了一些問題,這裡記錄一下,以備後查。
Android的JNI開發主要有兩中情況:一種是使用已經編譯好的.so動態庫;一種是使用c/c++原始碼開發。下面分別介紹:
一、使用已經編譯好的so
這種情況比較件簡單,只要把的.so檔案放到相應的目錄即可。如下:
.[module_name]
. . [src]
. . .[main]
. . . .[jniLibs]
. . . . .[armeabi]
. . . . .[armeabi-v7a]
. . . . .[x86]
. . . . .[mips]
注意 jniLibs
目錄是放在 module
下面,在Android
Studio中效果如下,這樣編譯之後so就會被自動打包進apk,程式碼中直接 loadLibrary即可了:
1 String libName = "helloNDK"; //庫名, 注意沒有字首lib和字尾.so 2 System.loadLibrary( libName );
二、使用C/C++原始碼
1 r9d以上版本NDK
首先確保自己的NDK版本在r9d以上,目前最新可以拿到的是r10,下載地址:
感謝 Android Studio中文組的無私奉獻。
如果低於r9d版本,Android studio 下ndk編譯會出現 No rule to make target 的錯誤。
2 配置 ndk.dir
在 local.properties
新增如下配置:
sdk.dir=/path/to/android-sdk
ndk.dir=/path/to/android-ndk
黃色部分修改為你本機的實際情況,如我的是:
sdk.dir=D\:\\android-sdk-windows
ndk.dir=D\:\\android-ndk
3 app下的build.gradle配置NDK模組
android { compileSdkVersion 19 buildToolsVersion "20.0.0" defaultConfig { applicationId "com.example.ndksample" minSdkVersion 9 targetSdkVersion 19 versionCode 1 versionName "1.0" ndk { moduleName "helloNDK" // <-- This is the name of my C++ module! } } // ... more gradle stuff here ... } // end of android section
ndk
還可以配置更多選項,如下:
ndk { moduleName "myEpicGameCode" cFlags "-DANDROID_NDK -D_DEBUG DNULL=0" // Define some macros ldLibs "EGL", "GLESv3", "dl", "log" // Link with these libraries! 在這裡新增你原先在makefile裡ldlibs所連結的庫 stl "stlport_shared" // Use shared stlport library }
4 新增C/C++原始碼
預設情況下,你需要把C/C++原始碼放在 [module]/src/main/jni/
路徑下。效果如下圖所示:
當然,也可以自定義原始碼路徑:
android { // .. android settings .. sourceSets.main { jni.srcDirs 'src/main/source' } }
這裡有一點需要注意,目前windows上android studio 下ndk編譯有一個bug,就是如果你的原始檔(.C和.cpp檔案,不算.h)只有一個,那麼最後也會有編譯error,目前google還沒有修復,暫時的解決辦法是,新建一個空的.c檔案放在那裡,就可以編譯通過了。。。。有點奇葩!!
5 分平臺配置編譯
這一步不是必須的,你可以根據需要,對各個平臺進行不同的編譯配置,可以設定覆蓋前面的編譯選項(例如 cFlags
)。例如你只想編譯指定平臺的.so,而不是所有的平臺。如下:
android { // .. android settings .. productFlavors { x86 { ndk { abiFilter "x86" } } arm { ndk { abiFilter "armeabi-v7a" } } mips { ndk { abiFilter "mips" } } } } // android
參考連結: