編輯:關於Android編程
因為工作需要可能要用到JNI開發,本篇文章就分享一下我在這方面的實踐,以前我們使用Eclipse做NDK開發,非常麻煩,需要配cygwin的編譯環境,後面NDK功能完善才逐漸簡單點
Eclipse的如何通過NDK生成so庫就不多說了,目前已經不適用於Android開發,建議使用AS進行開發,本篇也是基於AS來進行講解。
如果你進行Android開發一段時間還不知道什麼是JNI的話,說明你還是初學者,沒有理解Java層跟Native層之間是如何交互的。JNI(Java Native Interface的縮寫),Java層通過JNI來調用Native層的功能模塊,這樣做的好處是能消除平台的差異性,避免重復制造輪子。Java的跨平台性也體現在這裡。
如何通過JNI調用Native層的c/c++代碼,可以參考我的一篇文章:
JNI_最簡單的Java調用C/C++代碼
注:window的動態庫是.dll文件,而Linux下的動態庫是.so文件
如果翻不了牆,可以到這裡下載相應平台的NDK版本
http://www.androiddevtools.cn/
下載成功之後解壓縮,然後配置系統環境變量,拿windows舉例:
先新建NDK_HOME
然後再添加到PATH環境變量中<喎?/kf/ware/vc/" target="_blank" class="keylink">vcD4NCjxwPjxpbWcgYWx0PQ=="PATH環境變量" src="/uploadfile/Collfiles/20160429/20160429090331384.png" title="\" />
ok,配好之後,直接可以在命令行使用ndk-build命令:
這裡提示沒有定義NDK_PROJECT_PATH變量,暫且不管,我們後面說。
然後,在Android Studio配置NDK路徑:
上面是筆者的路徑,具體按你們來配。
至此,NDK環境配置完畢。
這裡創建一個Android項目-JNIDemo,然後定義TestJNI類,代碼如下:
package com.devilwwj.jnidemo;
/**
* Created by wwj_748 on 2016/4/27.15.47
*/
public class TestJNI {
public native boolean Init();
public native int Add(int x, int y);
public native void destory();
}
執行完上面的命令之後,就生成了com_devilwwj_jnidemo_TestJNI.h這個文件:
ok,這樣我們就可以進行下一步操作了。
然後根據.h文件,創建相對應的.cpp文件
com_devilwwj_jnidemo_TestJNI.cpp
//
// Created by wwj_748 on 2016/4/27.
//
#include
#include
#include "com_devilwwj_jnidemo_TestJNI.h"
#include "Add.h"
CAdd *pCAdd = NULL;
JNIEXPORT jboolean JNICALL Java_com_devilwwj_jnidemo_TestJNI_Init(JNIEnv *env, jobject obj) {
if (pCAdd == NULL) {
pCAdd = new CAdd;
}
return pCAdd != NULL;
}
JNIEXPORT jint JNICALL Java_com_devilwwj_jnidemo_TestJNI_Add
(JNIEnv *env, jobject obj, jint x, jint y) {
int res = -1;
if (pCAdd != NULL) {
res = pCAdd->Add(x, y);
}
return res;
}
JNIEXPORT void JNICALL Java_com_devilwwj_jnidemo_TestJNI_destory
(JNIEnv *env, jobject obj) {
if (pCAdd != NULL) {
pCAdd = NULL;
}
}
這裡我還需要創建兩個文件,CAdd.h和CAdd.cpp:
ok,到目前我們已經完成JNI層的實現了。
在jni目錄下,我們需要創建兩個mk文件
Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := JNIDemo
LOCAL_SRC_FILES := com_devilwwj_jnidemo_TestJNI.cpp
LOCAL_SRC_FILES += Add.cpp
include $(BUILD_SHARED_LIBRARY)
其中LOCAL_PATH是C/C++代碼所在目錄,也就是我們的jni目錄。
LOCAL_MODULE是要編譯的庫的名稱。編譯器會自動在前面加上lib,在後面加上.so。
LOCAL_SRC_FILES是要編譯的C/C++文件。
Application.mk
APP_ABI := all
表示生成所有平台的動態庫。
在defaultConfig下,配置ndk:
ndk {
moduleName "JNIDemo" // 生成的so名字
}
在android標簽內配置sourceSets:
sourceSets {
main {
jniLibs.srcDirs = ['libs']
}
}
生成的so文件都在src/main/libs目錄下。
完整的gradle配置,可以看源碼。
然後就可以在libs目錄下看到所有平台的so庫:
至此,完整的so庫實踐基本完畢,期間遇到任何問題均可留言,歡迎討論交流。
關於NDK開發so庫會有很多坑,本篇博客也只是引大家入門,我們在使用第三方的SDK的時候,就可能會用到別人提供的so庫,也可能會遇到使用so庫出現問題,有很多原因,可能是提供了不同的cpu架構的so庫,在其他平台出現的crash或者是其他問題,相信我們會遇到的問題別人也會遇到,這時候google一下也許能找到解決方案,最後祝生活愉快。
很多Android系統手機都或多或少出現過信號不穩的問題,其實很多時候不是手機自身的質量不佳,而是我們沒有對其進行合理的網絡配置。以聯通WCDMA制式的An
Tab與TabHost:這就是Tab,而盛放Tab的容器就是TabHost 。如何實現?? 每一個Tab還對應了一個布局,這個就有點好玩了。一個Activity,對應了多
前述:本人已工作兩年多,但是依然感覺還是Android的門外漢,之前一直從事Android的應用開發,每天就是各種調用SDK方法,各種拷貝網上的源碼以及jar包,從來也不
今天做一個小Demo用到了SoundPool,總結一下。MediaPlayer缺點:不能重疊播放音樂,封裝程度比較高,所以加載起來會比較慢。SoundPool簡介Andr