一.JNI和NDK介绍
JNI(Java Native Interface)是方便Java调用C、C++等Native代码所封装的一层接口,相当于一座桥梁。通过JNI可以操作一些Java无法完成的与系统相关的特性,尤其在图像和视频处理中大量用到。
NDK(Native Development Kit)是Google提供的一套工具,其中一个特性是提供了交叉编译,即C或者C++不是跨平台的,但通过NDK配置生成的动态库却可以兼容各个平台。源码通过NDK编译后可以生成在Android平台上运行的二进制文件.so及bin文件。
二.配置一下ndk环境
打开AndroidStudio, File--->Project Structure--->SDK Location--->Android NDK location 下加入ndk的本地路径。
三.创建流程
1.在/src/main/ 下创建一个JNI Folder,New--->Folder--->JNI Folder,名字自取。
2.在jni目录下,创建实现的c或c++文件,来实现后续h文件中的方法。
3.创建CmakeLists.txt,在app下与build.gradle同级,内容如下:
#标注需要支持的CMake最小版本
cmake_minimum_required(VERSION 3.4.1)
#add_library 定义需要编译的代码库 名称, 类型, 包含的源码
add_library(
# Sets the name of the library.
Test//首字母大写,要不提示错误
# Sets the library as a shared library.
SHARED
# source file
src/main/jni/test.cpp
)
#find_library 定义当前代码库需要依赖的系统或者第三方库文件(可以写多个)
find_library(
log_lib # 指定要查找的系统库, 给一个名字
log # 真正要查找的liblog.so或者liblog.a
)
# target_link_libraries设置最终编译的目标代码库
target_link_libraries(
Test # add_library 生成的
${log_lib} # find_library 找到的系统库
)
4.在build.gradle的android里面添加以下内容:
externalNativeBuild {
cmake {
path "CMakeLists.txt"
}
}
必要的话,在defaultConfig内部加入:
externalNativeBuild {
cmake {
cppFlags ""
abiFilters "armeabi-v7a", "arm64-v8a", "x86"
}
}
5.编写java文件,在/src/main/java/com/hly/learn/下创建包含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目录下,执行javah -encoding utf8 com.hly.learn.fragments.JniFragement(包含native方法的java类),会生成com_hly_learn_fragments_JniFragment.h文件,文件内容如下:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include
/* 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++文件中实现h中的方法,即步骤2,实现如下:
#include
//导入我们创建的头文件
#include "com_hly_learn_fragments_JniFragment.h"
JNIEXPORT jstring JNICALL Java_com_hly_learn_fragments_JniFragment_getFromJNI
(JNIEnv *env, jobject) {
//返回一个字符串
return env->NewStringUTF("Hello, my name is seven");
}
7.上层可以调用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会显示"Hello, my name is seven"
return view;
}
当进入界面时,TextView会显示"Hello, my name is seven"
8.C/C++函数调用Java函数
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++实现如下:
JNIEXPORT jstring JNICALL Java_com_hly_learn_fragments_JniFragment_setFromJava
(JNIEnv * env, jobject ) {
//通过反射获取上层的java类
jclass clazz = env->FindClass("com/hly/learn/fragments/JniFragment");
//实例化该类
jobject job = env->AllocObject(clazz);
//得到要调用类的方法
//参数列表:反射类,方法名称,(参数类型)返回类型
jmethodID jmeId = env->GetMethodID(clazz, "getFromJava", "()Ljava/lang/String;");
//调用方法
jstring result = (jstring)env->CallObjectMethod(job, jmeId);
return result;
}
当点击TextView时,会显示setFromJava()返回的字符串,setFromJava()在底层实现会调用java中的方法getFromJava(),从而最终Toast显示"This string is from java"。