Android 從Eclipse轉(zhuǎn)Android Studio的NDK配置方式

前言


之前一直是用Eclipse開發(fā)的想邦,后來轉(zhuǎn)AndroidStudio的時(shí)候遇到了一些坑支鸡,其中比較麻煩的就是NDK的編譯。

代碼已上傳至GitHub

myExample是我創(chuàng)建專門用來放小例子的,以后我寫文章需要的例子就都放這里面了垂攘。

正文


1.原理

之前用Eclipse的時(shí)候都是手動(dòng)創(chuàng)建一個(gè)jni文件夾菠劝,然后自己添加Android.mkApplication.mk2個(gè)文件赊舶,最后通過執(zhí)行ndk-build命令生成so庫。

但是AndroidStudio自帶編譯so庫功能赶诊,它會(huì)通過build.gradle里面的android.ndk項(xiàng)自動(dòng)生成Android.mk文件然后生成so庫笼平,我們需要做的就是禁止它自帶的功能,使用我們自己的mk文件舔痪。

實(shí)現(xiàn)步驟

  1. 禁止AndroidStudio自帶的ndk功能寓调。
  • 添加gradle task自動(dòng)調(diào)用ndk-build命令。

首先看下文件目錄

文件目錄

2.配置

1.禁止AndroidStudio自帶的ndk功能

完整配置請(qǐng)看build.gradle sourceSets

android {
    //... 省略其他配置
    sourceSets {
        main {
            jni.srcDirs = [] //禁止自帶的ndk功能
            jniLibs.srcDir 'src/main/libs' //重定向so目錄為src/main/libs
        }
    }
}
2.添加gradle task自動(dòng)調(diào)用ndk-build命令

完整配置請(qǐng)看build.gradle task

android {
    //... 省略其他配置

    tasks.withType(JavaCompile) {
        compileTask -> compileTask.dependsOn ndkBuild
    }

    task ndkBuild(type: Exec, description: 'Compile JNI source with NDK') {
        Properties properties = new Properties()
        properties.load(project.rootProject.file('local.properties').newDataInputStream())
        def ndkDir = properties.getProperty('ndk.dir')//獲得ndk目錄

        //執(zhí)行ndk-build命令
        if (Os.isFamily(Os.FAMILY_WINDOWS)) {//如果是win系統(tǒng)
            //commandLine "$ndkDir/ndk-build.cmd", 'NDK_DEBUG=1', '-C', file('src/main/jni').absolutePath //debug命令辙喂,如果需要做調(diào)試可以使用這條命令
            commandLine "$ndkDir/ndk-build.cmd", '-C', file('src/main/jni').absolutePath//這里指定了jni的目錄位置
        } else {
            //commandLine "$ndkDir/ndk-build", 'NDK_DEBUG=1', '-C', file('src/main/jni').absolutePath
            commandLine "$ndkDir/ndk-build", '-C', file('src/main/jni').absolutePath
        }
    }

    task ndkClean(type: Exec, description: 'Clean NDK Binaries') {
        Properties properties = new Properties()
        properties.load(project.rootProject.file('local.properties').newDataInputStream())
        def ndkDir = properties.getProperty('ndk.dir')//獲得ndk目錄

        //執(zhí)行ndk-build clear命令
        if (Os.isFamily(Os.FAMILY_WINDOWS)) {//如果是win系統(tǒng)
            commandLine "$ndkDir/ndk-build.cmd", 'clean', '-C', file('src/main/jni').absolutePath
        } else {
            commandLine "$ndkDir/ndk-build", 'clean', '-C', file('src/main/jni').absolutePath
        }
    }

    clean.dependsOn 'ndkClean'
}
  • 在生成so庫后可以把它注釋了捶牢,不然jni文件多了生成會(huì)比較慢。
  • 千萬別忘了在local.properties里配置ndk.dir不然會(huì)找不到ndk巍耗。

3.實(shí)現(xiàn)

build.gradle配置了上述語句后就可以自動(dòng)生成so庫了秋麸。

接下來讓我們來實(shí)現(xiàn)一個(gè)HelloJni試下。

1.創(chuàng)建Application.mk和Android.mk文件

Application.mk文件非常簡(jiǎn)單炬太,就是一句APP_ABI := all灸蟆,它的意思是生成所有架構(gòu)的so庫,在平常使用中我們一般會(huì)有選擇性的設(shè)置為APP_ABI := armeabi armeabi-v7a x86亲族。

Android.mk文件比較復(fù)雜炒考,推薦看《Android.mk、Application.mk》寫的比較詳細(xì)霎迫,我這里寫的比較簡(jiǎn)單斋枢。

LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)

LOCAL_MODULE    := HelloJni#模塊的名稱纱耻,會(huì)在生成的so庫前面加上lib斧蜕,最終名稱就是libHelloJni.so

LOCAL_SRC_FILES := HelloJni.cpp#要打包的源碼
include $(BUILD_SHARED_LIBRARY)
2.創(chuàng)建Java文件

我們創(chuàng)建一個(gè)HelloJni.java文件,用于ndk交互松捉,這個(gè)文件也特別簡(jiǎn)單涩赢,要注意的是我這里的hello()方法用的static戈次,在jni里對(duì)應(yīng)第二個(gè)參數(shù)為jclass,如果不是靜態(tài)方法筒扒,就會(huì)對(duì)應(yīng)jobject怯邪。

public class HelloJni {
    public static native String hello();
}
3.創(chuàng)建C++文件

首先需要?jiǎng)?chuàng)建.h文件,有二種創(chuàng)建方法花墩,第一種就是手動(dòng)創(chuàng)建悬秉,第二種就是利用javah.exe這個(gè)工具創(chuàng)建澄步,我們這里就利用第二種方法來創(chuàng)建這個(gè).h文件。

javah.exe這個(gè)工具在jdk的bin目錄下搂捧,我們打開AndroidStudio的Terminal窗口然后輸入javah.exe -d app/src/main/jni -cp app/src/main/java com.xiuyukeji.ndk_config.HelloJni回車后我們就能生成com_xiuyukeji_ndk_config_HelloJni.h文件驮俗。

這里有點(diǎn)需要注意,函數(shù)名對(duì)應(yīng)java層的包名+類名+方法名允跑,如果不是自動(dòng)生成王凑,要檢查下函數(shù)名是否對(duì)應(yīng),不然會(huì)報(bào)UnsatisfiedLinkError錯(cuò)誤聋丝。

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_xiuyukeji_ndk_config_HelloJni */

#ifndef _Included_com_xiuyukeji_ndk_config_HelloJni
#define _Included_com_xiuyukeji_ndk_config_HelloJni
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_xiuyukeji_ndk_config_HelloJni
 * Method:    hello
 * Signature: ()Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_xiuyukeji_ndk_1config_HelloJni_hello
  (JNIEnv *, jclass);

#ifdef __cplusplus
}
#endif
#endif
Terminal窗口
.H文件

我來解釋下javah.exe這條命令的意思索烹,-d代表輸出目錄在app/src/main/jni-cp代表加載目錄在app/src/main/java弱睦,最后的com.xiuyukeji.ndk_config.HelloJni就是文件名了百姓。

Javah工具說明
  • 如果提示找不到javah.exe就說明你的環(huán)境變量需要在path里多加一句%JAVA_HOME%\bin

輸出.h文件后我們就可以開始寫HelloJni.cpp了况木,代碼特別簡(jiǎn)單垒拢,返回一個(gè)字符串Hello jni,我這里加了個(gè)extern "C"火惊,意思是告訴編譯器按照C進(jìn)行編譯求类,不加也可以,具體解釋看這里《extern "c"用法解析》屹耐。

#include "com_xiuyukeji_ndk_config_HelloJni.h"

extern "C" {

    JNIEXPORT jstring JNICALL Java_com_xiuyukeji_ndk_1config_HelloJni_hello(JNIEnv * env, jclass jc){
        return (env)->NewStringUTF("Hello jni");
    }

}

然后我們可以開始生成so庫了尸疆,執(zhí)行Build->Build APK命令。

編譯命令
編譯過程

最終會(huì)在lib下生成各個(gè)架構(gòu)的so庫惶岭。

lib
4.調(diào)用So庫

上面我們已經(jīng)生成了so庫寿弱,下面我們就需要在java層調(diào)用libHelloJni.so里面的hello函數(shù)。

  1. MainActivity里動(dòng)態(tài)加載so庫按灶,一般加載so庫都是利用static {}症革,這種寫法叫靜態(tài)代碼塊優(yōu)先于其他代碼執(zhí)行。
static {
      System.loadLibrary("HelloJni");
}
public class MainActivity extends AppCompatActivity {
    //... 省略其他代碼
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);

        FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
        fab.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Snackbar.make(view, HelloJni.hello(), Snackbar.LENGTH_LONG).show();
            }
        });
    }
}
最終結(jié)果

結(jié)尾


如果它有解決你的問題的話地沮,請(qǐng)下點(diǎn)個(gè)贊,謝謝羡亩。

這是我個(gè)人的第三篇文章,寫于2017年4月19日危融。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末畏铆,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子吉殃,更是在濱河造成了極大的恐慌辞居,老刑警劉巖楷怒,帶你破解...
    沈念sama閱讀 206,378評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異瓦灶,居然都是意外死亡鸠删,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門贼陶,熙熙樓的掌柜王于貴愁眉苦臉地迎上來刃泡,“玉大人,你說我怎么就攤上這事碉怔『嫣” “怎么了?”我有些...
    開封第一講書人閱讀 152,702評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵撮胧,是天一觀的道長(zhǎng)桨踪。 經(jīng)常有香客問我,道長(zhǎng)芹啥,這世上最難降的妖魔是什么锻离? 我笑而不...
    開封第一講書人閱讀 55,259評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮墓怀,結(jié)果婚禮上汽纠,老公的妹妹穿的比我還像新娘。我一直安慰自己捺疼,他們只是感情好疏虫,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,263評(píng)論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著啤呼,像睡著了一般卧秘。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上官扣,一...
    開封第一講書人閱讀 49,036評(píng)論 1 285
  • 那天翅敌,我揣著相機(jī)與錄音,去河邊找鬼惕蹄。 笑死蚯涮,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的卖陵。 我是一名探鬼主播遭顶,決...
    沈念sama閱讀 38,349評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼泪蔫!你這毒婦竟也來了棒旗?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,979評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤撩荣,失蹤者是張志新(化名)和其女友劉穎铣揉,沒想到半個(gè)月后饶深,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,469評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡逛拱,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,938評(píng)論 2 323
  • 正文 我和宋清朗相戀三年敌厘,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片朽合。...
    茶點(diǎn)故事閱讀 38,059評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡俱两,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出旁舰,到底是詐尸還是另有隱情锋华,我是刑警寧澤,帶...
    沈念sama閱讀 33,703評(píng)論 4 323
  • 正文 年R本政府宣布箭窜,位于F島的核電站毯焕,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏磺樱。R本人自食惡果不足惜纳猫,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,257評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望竹捉。 院中可真熱鬧芜辕,春花似錦、人聲如沸块差。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽憨闰。三九已至状蜗,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間鹉动,已是汗流浹背轧坎。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留泽示,地道東北人缸血。 一個(gè)月前我還...
    沈念sama閱讀 45,501評(píng)論 2 354
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像械筛,于是被迫代替她去往敵國和親捎泻。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,792評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容