一.JNI和NDK介紹
???????JNI(Java Native Interface)是方便Java調(diào)用C达吞、C++等Native代碼所封裝的一層接口济赎,相當(dāng)于一座橋梁惊畏。通過JNI可以操作一些Java無法完成的與系統(tǒng)相關(guān)的特性扰法,尤其在圖像和視頻處理中大量用到轧飞。
???????NDK(Native Development Kit)是Google提供的一套工具衅鹿,其中一個(gè)特性是提供了交叉編譯,即C或者C++不是跨平臺的过咬,但通過NDK配置生成的動(dòng)態(tài)庫卻可以兼容各個(gè)平臺大渤。源碼通過NDK編譯后可以生成在Android平臺上運(yùn)行的二進(jìn)制文件.so及bin文件。
二.配置一下ndk環(huán)境
???????打開AndroidStudio掸绞, File--->Project Structure--->SDK Location--->Android NDK location 下加入ndk的本地路徑兼犯。
三.創(chuàng)建流程
1.在/src/main/ 下創(chuàng)建一個(gè)JNI Folder,New--->Folder--->JNI Folder集漾,名字自取切黔。
2.在jni目錄下,創(chuàng)建實(shí)現(xiàn)的c或c++文件具篇,來實(shí)現(xiàn)后續(xù)h文件中的方法纬霞。
3.創(chuàng)建CmakeLists.txt,在app下與build.gradle同級驱显,內(nèi)容如下:
#標(biāo)注需要支持的CMake最小版本
cmake_minimum_required(VERSION 3.4.1)
#add_library 定義需要編譯的代碼庫 名稱, 類型, 包含的源碼
add_library(
# Sets the name of the library.
Test//首字母大寫诗芜,要不提示錯(cuò)誤
# Sets the library as a shared library.
SHARED
# source file
src/main/jni/test.cpp
)
#find_library 定義當(dāng)前代碼庫需要依賴的系統(tǒng)或者第三方庫文件(可以寫多個(gè))
find_library(
log_lib # 指定要查找的系統(tǒng)庫, 給一個(gè)名字
log # 真正要查找的liblog.so或者liblog.a
)
# target_link_libraries設(shè)置最終編譯的目標(biāo)代碼庫
target_link_libraries(
Test # add_library 生成的
${log_lib} # find_library 找到的系統(tǒng)庫
)
4.在build.gradle的android里面添加以下內(nèi)容:
externalNativeBuild {
cmake {
path "CMakeLists.txt"
}
}
???????必要的話,在defaultConfig內(nèi)部加入:
externalNativeBuild {
cmake {
cppFlags ""
abiFilters "armeabi-v7a", "arm64-v8a", "x86"
}
}
5.編寫java文件埃疫,在/src/main/java/com/hly/learn/下創(chuàng)建包含native方法的類java文件:
public class JniFragment extends Fragment {
static {
System.loadLibrary("Test");
}
public native String getFromJNI();
@Override
public void onAttach(Context context) {
super.onAttach(context);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View view = inflater.inflate(getLayoutId(), container, false);
return view;
}
}
6.在/src/main/java目錄下伏恐,執(zhí)行javah -encoding utf8 com.hly.learn.fragments.JniFragement(包含native方法的java類),會(huì)生成com_hly_learn_fragments_JniFragment.h文件栓霜,文件內(nèi)容如下:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_hly_learn_fragments_JniFragment */
#ifndef _Included_com_hly_learn_fragments_JniFragment
#define _Included_com_hly_learn_fragments_JniFragment
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_hly_learn_fragments_JniFragment
* Method: getFromJNI
* Signature: ()Ljava/lang/String;
*/
JNIEXPORT jstring JNICALL Java_com_hly_learn_fragments_JniFragment_getFromJNI
(JNIEnv *env, jobject);
#ifdef __cplusplus
}
#endif
#endif
然后把com_hly_learn_fragments_JniFragment.h mv 到 jni目錄下翠桦,然后在c或c++文件中實(shí)現(xiàn)h中的方法,即步驟2,實(shí)現(xiàn)如下:
#include<jni.h>
//導(dǎo)入我們創(chuàng)建的頭文件
#include "com_hly_learn_fragments_JniFragment.h"
JNIEXPORT jstring JNICALL Java_com_hly_learn_fragments_JniFragment_getFromJNI
(JNIEnv *env, jobject) {
//返回一個(gè)字符串
return env->NewStringUTF("Hello, my name is seven");
}
7.上層可以調(diào)用native方法了
public class JniFragment extends Fragment {
static {
System.loadLibrary("Test");
}
public native String getFromJNI();
@Override
public void onAttach(Context context) {
super.onAttach(context);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View view = inflater.inflate(getLayoutId(), container, false);
TextView tv = view.findViewById(R.id.tv);
tv.setText(getFromJNI());//TextView會(huì)顯示"Hello, my name is seven"
return view;
}
???????當(dāng)進(jìn)入界面時(shí)销凑,TextView會(huì)顯示"Hello, my name is seven"
8.C/C++函數(shù)調(diào)用Java函數(shù)
???????java方法如下:
public class JniFragment extends BaseFragment {
static {
System.loadLibrary("Test");
}
public native String getFromJNI();
public native String setFromJava();
public String getFromJava() {
return "This string is from java";
}
@Override
public int getLayoutId() {
return R.layout.jni_layout;
}
@Override
public void initData(View view) {
TextView tv = view.findViewById(R.id.tv);
tv.setText(getFromJNI());
tv.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(mContext, setFromJava(), Toast.LENGTH_SHORT).show();
}
});
}
}
???????C/C++實(shí)現(xiàn)如下:
JNIEXPORT jstring JNICALL Java_com_hly_learn_fragments_JniFragment_setFromJava
(JNIEnv * env, jobject ) {
//通過反射獲取上層的java類
jclass clazz = env->FindClass("com/hly/learn/fragments/JniFragment");
//實(shí)例化該類
jobject job = env->AllocObject(clazz);
//得到要調(diào)用類的方法
//參數(shù)列表:反射類丛晌,方法名稱,(參數(shù)類型)返回類型
jmethodID jmeId = env->GetMethodID(clazz, "getFromJava", "()Ljava/lang/String;");
//調(diào)用方法
jstring result = (jstring)env->CallObjectMethod(job, jmeId);
return result;
}
???????當(dāng)點(diǎn)擊TextView時(shí)斗幼,會(huì)顯示setFromJava()返回的字符串澎蛛,setFromJava()在底層實(shí)現(xiàn)會(huì)調(diào)用java中的方法getFromJava(),從而最終Toast顯示"This string is from java"蜕窿。