http://www.open-open.com/lib/view/open1451917048573.htmlhtml
Java Native Interface (JNI)標準是java平臺的一部分,它容許Java代碼和其餘語言寫的代碼進行交互。JNI 是本地編程接口,它使得在 Java 虛擬機 (VM) 內部運行的 Java 代碼可以與用其它編程語言(如 C、C++ 和彙編語言)編寫的應用程序和庫進行交互操做。
因爲Android的應用層的類都是以Java寫的,這些Java類編譯爲Dex型式的Bytecode以後,必須靠Dalvik虛擬機(VM: Virtual Machine)來執行。VM在Android平臺裏,扮演很重要的角色。
(多的咱不說了,介紹什麼的 處處都有能夠去搜一下)
我們先建一個NdkJniDemo的工程java
新建工程.pngandroid
新建JniUtils類實現native方法git
public class JniUtils { public static native String getStringFormC(); }
build.pnggithub
而後clean project 再rebuild project 生成class文件,
這時候打開以下圖的文件夾看是否生成了classes文件夾,沒有生成請從新來過。shell
文件夾.png編程
再打開Terminal輸入指令cd app/build/intermediates/classes/debug
而後再輸入指令javah -jni com.wobiancao.ndkjnidemo.ndk.JniUtils
注意 這裏javah -jni後面跟的是JniUtils類的全路徑,若是javah報不存在之類的,是你的java環境沒有配置好。android-studio
Paste_Image.pngruby
這時候打開classes/debug下面的文件發現多了一個文件com_wobiancao_ndkjnidemo_ndk_JniUtils.h
而後在src/main下新建文件夾jni,把生成的.h文件複製或者剪切到jni文件夾下面去,新建一個c類隨便取一個名字,添加代碼以下bash
//// Created by XY on 16/1/4.//#include "com_wobiancao_ndkjnidemo_ndk_JniUtils.h"/* * Class: Java_com_wobiancao_ndkjnidemo_ndk_JniUtils * Method: getStringFormC * Signature: ()Ljava/lang/String; */JNIEXPORT jstring JNICALL Java_com_wobiancao_ndkjnidemo_ndk_JniUtils_getStringFormC (JNIEnv *env, jobject obj){ return (*env)->NewStringUTF(env,"這裏是來自c的string");
這裏發現頭文件#include <jni.h>
報紅色,是由於我們尚未配置ndk環境,打開file->project structure
選擇你所下載的ndk環境路徑,若是沒有ndk這裏有個地址你們能夠去下載 一個安卓工具集合的網站:http://androiddevtools.cn/
Paste_Image.png
設置好了以後,發現頭文件仍是紅色的,而後再build一下工程,就會有提示
Error: NDK integration is deprecated in the current plugin. Consider trying the new experimental plugin. For details, see http://tools.android.com/tech-docs/new-build-system/gradle-experimental. Set "android.useDeprecatedNdk=true" in gradle.properties to continue using the current NDK integration.
按着提示作就好了 在gradle.properties文件末尾添加android.useDeprecatedNdk=true
就ok啦
而後在app文件下得build.gradle ->defaultConfig括號內添加以下代碼
ndk { moduleName "NdkJniDemo" //生成的so名字 abiFilters "armeabi", "armeabi-v7a", "x86" //輸出指定三種abi體系結構下的so庫,目前無關緊要。}
到了這一步從新build項目,發現已經沒有變紅了。接下來就是運用了,在JniUtils類裏面添加以下代碼
static { System.loadLibrary("NdkJniDemo");//以前在build.gradle裏面設置的so名字,必須一致}
而後簡單調用就好了,MainActivity代碼以下
public class MainActivity extends AppCompatActivity { TextView textView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); textView = (TextView) findViewById(R.id.ndk_text); textView.setText(JniUtils.getStringFormC()); } }
運行結果如圖
運行圖.png
我們打開app->intermediates-ndk-debug發現生成了三個文件夾,而且對應了以前咱們在build.gradle配置的abiFilters
大功告成。
新建libs文件夾把這三個文件夾放進去
Paste_Image.png
而後刪除我們的jni文件試試,是否工程還能運用?
ok 教程完畢
最後出一個利用jni的Aes加密demo運行結果如圖所示
Aes加密.png
項目地址:https://github.com/a12a15a05/NdkJniDemo
謝謝star歡迎點贊=。=轉載請註明來源
來自: http://www.jianshu.com/p/d8cde65cb4f7
http://blog.csdn.net/enlangs/article/details/51881718
AS的NDK插件不須要編寫Application.mk和Android.mk,因此本身編寫的無效!!,關鍵是那插件功能太弱了(不少Android.mk的屬性不能修改)。若是想用本身的mk文件,須要在app/build.gradle中打掉NDK的全部配置。
android {
compileSdkVersion 23 buildToolsVersion "23.0.2" defaultConfig { applicationId "ckm.xxx.testndk" minSdkVersion 19 targetSdkVersion 23 versionCode 1 versionName "1.0" // 若是要使用本身的Application.mk和Android.mk須要修改下面2處。 // 修改1,直接打掉下面配置 // ndk{ // moduleName "test" // abiFilters "armeabi","armeabi-v7a" // } } buildTypes { release { minifyEnabled false proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } } // sourceSets { main { jni.srcDirs = ['src/main/jni'] } } // 修改2,須要把jni.srcDirs設置爲空。 // sourceSets { // main { // jni.srcDirs = [] // } // } }
自動生成頭文件,在extend tool中增長配置:
參數內容:
Program:C:\Program Files\Java\jdk1.8.0_25\bin\javah.exe Parameters:-v -jni -d $ModuleFileDir$\src\main\jni $FileClass$ Working directory:$SourcepathEntry$
設置結果
- 設置ndk-build的extend tool
Program:C:\ndk\ndk-build.cmd Parameters:NDK_PROJECT_PATH=$ModuleFileDir$/build/intermediates/ndk NDK_LIBS_OUT=$ModuleFileDir$/src/main/jniLibs NDK_APPLICATION_MK=$ModuleFileDir$/src/main/jni/Application.mk APP_BUILD_SCRIPT=$ModuleFileDir$/src/main/jni/Android.mk V=1 Working directory:$SourcepathEntry$
最終結果:
疑惑:
注意:
若是按照上面的方式使用自定義Android.mk那麼AS將沒法識別到native函數。
來源:http://stackoverflow.com/questions/21096819/jni-and-gradle-in-android-studio
此方案主要是直接修改build.gradle
apply plugin: 'com.android.application' android { compileSdkVersion 14 buildToolsVersion "20.0.0" defaultConfig { applicationId "com.example.application" minSdkVersion 14 targetSdkVersion 14 ndk { moduleName "YourModuleName" } } sourceSets.main { jni.srcDirs = [] // This prevents the auto generation of Android.mk jniLibs.srcDir 'src/main/libs' // This is not necessary unless you have precompiled libraries in your project. } task buildNative(type: Exec, description: 'Compile JNI source via NDK') { def ndkDir = android.ndkDirectory commandLine "$ndkDir/ndk-build", '-C', file('src/main/jni').absolutePath, // Change src/main/jni the relative path to your jni source '-j', Runtime.runtime.availableProcessors(), 'all', 'NDK_DEBUG=1' } task cleanNative(type: Exec, description: 'Clean JNI object files') { def ndkDir = android.ndkDirectory commandLine "$ndkDir/ndk-build", '-C', file('src/main/jni').absolutePath, // Change src/main/jni the relative path to your jni source 'clean' } clean.dependsOn 'cleanNative' tasks.withType(JavaCompile) { compileTask -> compileTask.dependsOn buildNative } } dependencies { compile 'com.android.support:support-v4:20.0.0' }
參考:圖片和extend tool設置都來自這位博主,很是感謝。
參考(請同事寫了段代碼處理下):
#!/bin/sh
for arg in "$@" do if [[ $arg == NDK_PROJECT_PATH=* ]];then param1=`echo $arg | awk -F = '{printf $1}'` elif [[ $arg == APP_BUILD_SCRIPT=* ]];then param2=`echo $arg | awk -F = '{printf $2}'` else param="$param $arg" fi done #echo $param2 param2=`echo $param2 | sed 's#build/intermediates/ndk/debug/Android.mk#src/main#g'` #echo $param2 #echo "" if [[ -n "$param1" ]] && [[ -n "$param2" ]] then param1="$param1=$param2" param="$param1 $param" fi DIR="$(cd "$(dirname "$0")" && pwd)" $DIR/build/ndk-build $param