1.编写一个如下类
(方法名报红可以不用管)
public class JniUtils {
public static native String getStringFromC();
public static native int add(int a ,int b);
}
2.rebuild项目后,在build/intermediates/classes下有编译后的class文件,Terminal命令行中:
cd 到那个JniUtils.class文件完整包名的上一目录,即到...build/intermediates/classes/debug目录处
执行: javah -jni 完整包名.类名 ( javah -jni com.emagroup.ethesdk.common.JniUtils)
3. 成功后 会在对应目录中生成一个.h文件, 接下来在项目中创建一个jni文件夹(和java文件夹平级), 将.h文件拷入.
我们先来查看一下这个.h文件的内容。这里面用java的概念来说就相当于接口内的抽象方法,需要我们创建.c文件来实现这些方法同时也就将我们的定义的native方法实现了
4.我们在jni目录下创建一个etheValue_jni.c
文件来实现.h文件中的抽象方法
//自己创建一个c文件,实现自己定义的native方法,也就是.h文件中的方法
//引入自己生成的.h头文件
#include <com_emagroup_ethesdk_common_JniUtils.h>
//返回一个字符串
JNIEXPORT jstring JNICALL Java_com_emagroup_ethesdk_common_JniUtils_getStringFromC
(JNIEnv *env, jclass jobj) {
return (*env)->NewStringUTF(env,"HelloWorld 我是用jni调用出来的字符串");
}
//返回 a+b的结果
JNIEXPORT jint JNICALL Java_com_emagroup_ethesdk_common_JniUtils_add
(JNIEnv *env, jclass jobj, jint a, jint b){
return a+b;
}
5. 接下来我们在build.gradle
中添加ndk配置
defaultConfig {
//...
ndk {
moduleName "etheso"//指定生成的so文件名
abiFilters "armeabi", "armeabi-v7a", "x86"//cpu的类型
}
}
6. 将项目rebuild过后我们在./build就能看到生成的so文件了
(如果中间出现 jni your project contain c++... cmake or ndk-build 的问题,
在gradle.properties中加上 )
android.useDeprecatedNdk = true
7. 现在就可以调用了
static {
//名字必须和build.gradle中的moduleName一致
System.loadLibrary("etheso");
}
......
L.e("jniString",JniUtils.getStringFromC());
L.e("JniAdd","value:"+JniUtils.add(1,2));