如果你进行Android开发一段时间还不知道什么是JNI的话,说明你还是初学者,没有理解Java层跟Native层之间是如何交互的。JNI(Java Native Interface的缩写),Java层通过JNI来调用Native层的功能模块,这样做的好处是能消除平台的差异性,避免重复制造*。Java的跨平台性也体现在这里。
如何通过JNI调用Native层的c/c++代码,可以参考我的一篇文章:
注:window的动态库是.dll文件,而Linux下的动态库是.so文件
配置NDK
=====
如果翻不了墙,可以到这里下载相应平台的NDK版本
http://www.androiddevtools.cn/
下载成功之后解压缩,然后配置系统环境变量,拿windows举例:
先新建NDK_HOME
然后再添加到PATH环境变量中
ok,配好之后,直接可以在命令行使用ndk-build命令:
这里提示没有定义NDK_PROJECT_PATH变量,暂且不管,我们后面说。
然后,在Android Studio配置NDK路径:
上面是笔者的路径,具体按你们来配。
至此,NDK环境配置完毕。
定义Native方法
==========
这里创建一个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();
}
使用javah命令生成.h文件
===============
执行完上面的命令之后,就生成了com_devilwwj_jnidemo_TestJNI.h这个文件:
ok,这样我们就可以进行下一步操作了。
创建jni目录,创建.cpp文件
================
然后根据.h文件,创建相对应的.cpp文件
com_devilwwj_jnidemo_TestJNI.cpp
//
// Created by wwj_748 on 2016/4/27.
//
#include <stdio.h>
#include <stdlib.h>
#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层的实现了。
创建Android.mk和Application.mk文件
=============================
在jni目录下,我们需要创建两个mk文件
Android.mk
LOCAL_PATH :=
《Android学习笔记总结+最新移动架构视频+大厂安卓面试真题+项目实战源码讲义》
【docs.qq.com/doc/DSkNLaERkbnFoS0ZF】 完整内容开源分享
$(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