使用NDK開發(fā)有件比較麻煩的事情就是編寫Application.mk
和Android.mk
的,而Android Studio 的插件gradle-experimental
就是用來解決這個(gè)這個(gè)問題的。使用gradle-experimental
插件可以不用再編寫*.mk
文件的情況下進(jìn)行NDK開發(fā)。
gradle-experimental
是Android Studio的一個(gè)實(shí)驗(yàn)性的項(xiàng)目,是基于gradle的一個(gè)插件,主要用來自動化NDK的配置實(shí)現(xiàn),無需自己編寫Application.mk
和Android.mk
,對于調(diào)試NDK項(xiàng)目也更加的友好,支持對于NDK的Debug。
下面就來嘗試下gradle-experimental
的便利吧!
環(huán)境要求:
- Android Studio > 2.0
- gradle>2.10
- Android NDK r10e
- Build Tools > 19.0.0
配置gradle-experimental
- 添加
gradle-experimental
依賴
在項(xiàng)目的主目錄下./build.gradle
中替換掉以前的build tools
,使用最新版本的gradle-experimental
buildscript {
repositories {
jcenter()
}
dependencies {
classpath "com.android.tools.build:gradle-experimental:0.7.0"
}
}
- 替換插件(plugin)
使用gradle-experimental
就不能再使用com.android.application
了,需要使用com.android.model.application
替代。
在項(xiàng)目主目錄下的./app/build.gradle
中更改plugin
apply plugin: "com.android.model.application"
- 配置model{}
增加model{}
,android的配置在model{}
中。
model {
android {
compileSdkVersion 23
buildToolsVersion "25.0.0"
ndk {
moduleName "experiment"
//stl = 'gnustl_static'
//toolchain = 'clang'
//todo 指定cpu
abiFilters.addAll(['armeabi', 'armeabi-v7a']) //this is default
ldLibs.addAll([ 'log']);
}
defaultConfig {
applicationId "com.jjz"
//需要使用.apiLevel
minSdkVersion.apiLevel 15
targetSdkVersion.apiLevel 23
versionCode 1
versionName "1.0"
}
buildTypes {
release {
minifyEnabled false
}
}
}
}
配置NDK
在配置NDK之前需要確認(rèn):
- NDK包是否下載
- NDK的環(huán)境變量是否配置
能夠正確運(yùn)行命令:ndk-build
。
以上配置完成之后,在./app/build.gradle
中配置要配置android.ndk
的相關(guān)內(nèi)容:
model {
android {
compileSdkVersion 23
buildToolsVersion "23.0.2"
ndk{
moduleName "experiment"
//toolchain = 'clang'
abiFilters.addAll(['armeabi', 'armeabi-v7a']) //this is default
ldLibs.addAll([ 'log']);
}
}
}
這里定義的moduleName
就是后面生成的NDK的包名。我們還可以指定其編譯成對應(yīng)的cpu,編譯工具,使用的類庫等。
源代碼配置
默認(rèn)情況下,在src/main/jni
下使用的是c/c++文件。也可以在model.android
中指定:
model {
android {
compileSdkVersion 23
buildToolsVersion "23.0.2"
ndk {
moduleName "experiment"
}
sources {
main {
jni {
source {
srcDir "src/main/jni"
}
}
java{
source{
srcDir "src/main/java"
}
}
}
}
}
}
在sources
分別指定了java
源代碼和jni
源代碼的目錄。
以上的配置完成之后,就可以使用gradle experimental
開始NDK開發(fā)了。
使用gradle experimental
在java里面定義一個(gè)native
方法,native標(biāo)識的方法會需要在jni中實(shí)現(xiàn),可以在java中調(diào)用,還需要加載NDK生成的.so
類庫。
public class NativeUtil {
//加載類庫
static {
System.loadLibrary("experiment");
}
public static native String firstNative();
}
這個(gè)時(shí)候編譯器會出現(xiàn)一個(gè)紅色的警告,因?yàn)槎x的native
方法編譯器并沒有找到對應(yīng)的實(shí)現(xiàn):
可以使用alt+enter
鍵可以自動生成jni文件:
注意這個(gè)時(shí)候生成直接的是experiment.c
,沒有.h
文件,生成的jni文件的內(nèi)容:
**#include <jni.h>**
JNIEXPORT jstring JNICALL
Java_com_jjz_NativeUtil_firstNative(JNIEnv *env, jclass type) {
// TODO
return (*env)->NewStringUTF(env, returnValue);
}
自動生成的jni文件并沒有具體的實(shí)現(xiàn),需要修改jni文件讓其返回一段測試內(nèi)容:
>JNIEXPORT jstring JNICALL
Java_com_jjz_NativeUtil_firstNative(JNIEnv **env, jclass type) {
char chars[] = "i am test";
return (*env)->NewStringUTF(env, chars);
}
這樣就完成了一個(gè)JNI開發(fā)調(diào)用,定義了一個(gè)native
方法,在.c
文件中對其進(jìn)行了實(shí)現(xiàn)。如果沒有gradle experimental
插件的話,現(xiàn)在就需要使用ndk-build
命令編譯成.so
文件,然后在運(yùn)行Android項(xiàng)目,調(diào)用方法看下是否能夠調(diào)用成功,而現(xiàn)在有了gradle experimental
就變的了如此簡單,下一步就需要直接運(yùn)行
就可以了,對!就是直接運(yùn)行,省略了中間的ndk-build
環(huán)節(jié),就和編寫Java代碼一樣,直接運(yùn)行
即可。
直接運(yùn)行
打開Toolbar
中的運(yùn)行配置菜單,可以看到:
除了
app
以外,多出了一個(gè)app-native
的菜單,這個(gè)選項(xiàng)可以直接編譯NDK源碼之后再運(yùn)行Android
,這樣編寫完NDK
之后可以運(yùn)行了,不需要再進(jìn)行ndk-build
:在Java中調(diào)用
NativeUtil.firstNative
,會得到j(luò)ni中寫好的返回值:i am test
,在這個(gè)例子中,我讓這個(gè)這字段顯示在主頁面上。
可以看到使用gradle-experimental
開發(fā)NDK,不需要再做復(fù)雜的配置,自動化native方法,不需要自定義.h頭文件,對開發(fā)更加友好。
文中源代碼地址:https://github.com/jjz/android/tree/master/experimental
·