源地址:http://www.voidcn.com/blog/chengkaizone/article/p-5761016.html
好記性不如爛筆頭,開始堅持寫博客,學一點記一點,只為了生活更好。
學了一年多的硬件,現在來做android,不知道是對是錯,跟着感覺走,開始總是想把android和硬件掛上勾,所以第一個android應用寫的是關於藍牙的。現在學習NDK開發,有時說得也是JNI開發。那為什么要學習JNI,JNI會學習到比較底層的東西,java可以實現上層應用的開發,但是不能寫底層驅動,有些android項目必須和底層硬件打交道,比如要精確快速的獲取某種傳感器的值,那么java是不能辦到的,那怎么辦,就通過JNI開發,通過c/c++完成底層驅動或者值得獲取,然后將值傳回到java層。同樣java層也可以將數據傳遞給c/c++層,這樣通過JNI完成數據的交互。
那為什么java不能訪問底層呢?先看一張java程序的執行流程圖:
java源代碼通過java編譯器后變成字節碼,然后裝載到java平台運行期環境(java虛擬機),在不同的平台下游不同的java虛擬機,window下有window的,linux下有linux下的java虛擬機,java虛擬機屏蔽了與底層直接的細節,做到java運行與平台無關,所以java是不能訪問底層的
那么java層怎么就能訪問到c/c++層並將數據傳輸過去:c/c++又怎么能將數據傳回java層呢。出來在java層通過native標記某個方法是本地方法外,重要是需要NDK這個android 本地開發工具集
什么是NDK(android native develop kits ):android 本地開發工具集 ,可以把c/c++ ->編譯成一個 linux下可以執行的二進制文件 java代碼里面就可以通過jni 調用執行二進制的文件.
什么是JNI :java本地開發接口,JNI是一個協議這個協議用來溝通java代碼和外部的本地代碼(c/c++).通過這個協議,java代碼就可以調用外部的c/c++,代碼外部的c/c++代碼也可以調用java代碼。
JNI開發用途:驅動開發 (wifi-hotspot) 2.3無線熱點共享 ,Native code效率高,數學運算,實時渲染的游戲上,音視頻處理(極品飛車,opengl,ffmpeg),復用代碼(文件壓縮,人臉識別…)等。
下面進入主題:JNI開發環境配置及簡單的程序實現。
一:需要的工具及資源:開發工具,android studio,NDK開發工具集,至於沒有的可以到網上下載,我用的是android studio1.3和 android-ndk-r10,當然不必和我一樣的,好像聽網上說android studio1.3開始支持DNK開發,而且支持也不太好的,經過這幾天的學習,真的是感覺支持不是很好的,最新的NDK開發工具集集成了cygwin(一個模擬Linux運行環境的軟件),所以就不像以前那樣還要安裝什么cygwin。
環境配置步驟如下(簡單的說一下,網上有很多教程):
1 解壓NDK開發工具集,隨便哪里都行,看你自己的習慣,但是還是要自己知道,別什么時候不知不覺刪了,然后開發JNI始終不行,怎么都找不到錯誤,那就杯具了。
2打開android studio,依次點擊:File ->ProjectStructure:如圖:
點擊右邊,選擇你解壓的NDK,點擊應用。這一步將工具包關聯到android studio。
在 local.properties 文件中設置ndk的路徑:
就是你解壓的NDK工具集的路徑 我的是 E:/Android/android-ndk-r10,注意要對應加斜杠
到這里我說JNI(有時說的NDK開發,一樣的,叫法不同)環境初步建好了,不想eclipse那樣什么cygwin啊,怎么還要ndk-build的啊,我說什么都不用,就用一個命令就行了,下面就開始我們的第一個小demo。
程序過程講解:
一:頭文件生成:
第一步:像平時一樣將一個空的android工程。
第二部:前面說了,要用native來標識一個方法,告訴程序這是一個本地方法。程序如下
public class NDK extends Activity{ static { System.loadLibrary("MyJni");//導入生成的鏈接庫文件 } public native String getStringFromNative();//本地方法 public native String getString_From_c(); protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.myjni); System.out.println(getStringFromNative()); } public void onClick(View view) { System.out.println(getString_From_c()); Toast.makeText(this, getStringFromNative(), Toast.LENGTH_LONG).show(); }}
這一步可以先不加這行代碼:System.loadLibrary("MyJni");//導入生成的鏈接庫文件 ,因為這里還沒有建立C文件,不能生成庫文件。
然后make project一下,目的就是編譯成對應的class文件。然后根據生成的class文件,利用javah生成對應的 .h頭文件。如果沒有編譯就執行javah命令會提示找不到這個類文件的。下面是我的工程界面;
這里說一下:我工程里面的本地方法顯示是紅色,我一開始以為不能進行JNI開發,或者有錯誤,包括我后面的C文件和頭文件里面也有很多地方是紅色的,我開始定義這些方法的時候是紅色的,但是有時又不是的,不知道是android studio對NDK支持不太好,還是怎么得,但是后面開發JNI程序沒有任何影響的。所以就沒有關了,如果那位大神知道還望告訴我一下,將非常感激。
第三部:打開命令窗口,點擊view ->ToolsWindows->Terminal,如下圖:
進來后默認是指向當前的工程目錄,接下來輸入命令:cd app/src/main 回車,切換到main目錄下:如圖:
至於怎么理解命令,建議你去學一下linux系統,這事linux最基本的操作。
第四部:輸入javah -d jni -classpath E:/Android/sdk/platforms/android-22/android.jar;../../build/intermediates/classes/debug example.daosong.com.ndkdemo.NDK(注意前后有英文的;號隔開的哈) 生成頭文件(要先編譯程序的,不然會報錯)。可能有些人看到這里就茫然了,這么長怎么寫出來的喲,這怎么記得到的,我告訴大家根本不用記,我給你說怎么得來的,你們就能很快寫出來了,是不是真的喲,不信試一試的。
首先:javah是生成頭文件需要的工具,這個很好記得把,相信學java是都用過的。-d jni 在工程下生成jni目錄,到時會在這個目錄下建JNI開始的C/C++源文件的。
-classpath E:/Android/sdk/platforms/android-22/android.jar 這個就是你SDK文件下android.jar所在的文件位置,找到后復制即可。在學java的時候講了這個的,可以將E:/Android/sdk/platforms/android-22/android.jar配置到環境變量就可以不寫出來,因為在生成頭文件是需要這個jar包,因為我沒有配置到環境變量,所以這里就顯示寫出來。這部分很好搞定把。前面好弄,那后面部分怎了弄得,相信學過linux的很快能夠知道的。
../../build/intermediates/classes/debug example.daosong.com.ndkdemo.NDK
首先是 ../../build/intermediates/classes/debug,結合到下面的圖看:
..表示返回上級目錄,../..表示返回上兩級目錄,也就是返回到那層目錄,然后明白這個了啥../../build/intermediates/classes/debug,指向debug目錄,然后空格和后面隔離,
然后就是后面部分example.daosong.com.ndkdemo.NDK,看上圖明白啥,是該類的全路徑,這里有.隔離開,平時寫類的全路徑都是用的.號,生成頭文件是將類對應的class文件生成二進制文件,所以要指向這里的,然后回車就能完成了頭文件的生成。意思是將該文件目錄下的NDK.class文件生成相應的頭文件。打開jni目錄下會有本地方法相應的映射方法定義。
在jni目錄下建立C源文件
網上說需要在jni下多建一個空的C文件,不讓會出錯,到底出什么錯我也不知道的,反正我多不多建都沒有什么影響的,不過多建一個沒什么影響,也就建了。
jni.c
代碼如下:
#include "example_daosong_com_ndkdemo_NDK.h"//#include <android/log.h>//#define LOG_TAG "System.out"//#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, __VA_ARGS__)//#define LOGINFO(...) __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__)JNIEXPORT jstring JNICALL Java_example_daosong_com_ndkdemo_NDK_getStringFromNative (JNIEnv * env, jobject obj){ // LOGINFO("LOGINFO"); return (*env)->NewStringUTF(env,"NDK 測試成功");}jstring Java_example_daosong_com_ndkdemo_NDK_getString_1From_1c (JNIEnv * env, jobject jobject){ return (*(*env)).NewStringUTF(env,"NDK 來自於C文件");}
講到這里如果要運行還得配置一個地方:
現在運行控制台就會打印相應的消息(至於為什么這么寫,代碼代表什么意思,我會在下一篇博客講解),不是eclipse還要配置android.mk這么文件嗎?android studio不用,你還真猜對了,不過不是說不用就代表他沒有,只不過這個配置過程不過你來做,你要做的就是配置上圖的代碼。那android.mk
在哪里的呢?看下圖:
是不是是曾相識啊,對就是它。不知不覺就是深夜1點多了,下面給出我的工程,里面有log打印相關的知識,這個將在下一篇博客詳細講解,如果有什么錯誤,歡迎大家指出。
http://download.csdn.net/detail/tuoguang/9068899
另外有一點很關鍵
如果出現如下錯誤: NDK integration is deprecated in the current plugin.
gradle.properties中添加如下配置
android.useDeprecatedNdk=true
來源地址: http://www.th7.cn/Program/Android/201509/550864.shtml