本文是AndroidJNI简单入门,大佬们请绕行(指点一二)…
定义
native 即 JNI,Java Native Interface,就是用户和本地C代码进行互操作的API
基本用法
在基本用法之前,要安装SDKTools
创建一个自己的Application,
然后IDE会自动创建一些文件,如图
在app moudle下有一个CMakeLists.txt文件,此文件是CMake 构建脚本,名字不可更改,路径可以随意换,但是必须和build.gradle中的path保持一致。
其中add_library()中是制定需要生成的.so库的名字,这里demo是native-lib。
后面还有设置source file的,比如你要引用的其他c++/c代码。把具体某路径下的某文件名字添加上。
gradle文件的变化如图:
abiFilters可以设定生成哪几种abi,不写就是生成所有类型的
externalNativeBuild {
cmake {
cppFlags ""
abiFilters 'armeabi','armeabi-v7a','x86'
}
}
指定你的CMakeLists文件的位置
android中
externalNativeBuild {
cmake {
path "CMakeLists.txt"
}
}
这时候运行APP,就会看到初步效果了。
但但但但是,我们往往在使用时需要多个c++/c的调用完成相关功能,但是对外的接口却只有一个类(native-lib.cpp),这就涉及到如何在native-lib.cpp中调用其他c++/c的代码了。在此demo的基础上做下扩展。
OK,开始正文
1.创建一个jni库方法的调用类 JniTest.java
/**
* A native method that is implemented by the 'native-lib' native library,
* which is packaged with this application.
*/
static {
System.loadLibrary("native-lib");
}
public static native String stringFromJNIString();
public static native String Encode(String parm);
2.第二步在在cpp之下new c++ class(当然这部分代码是C++工程师写的,这里我就关公面前耍大刀,进行最简单实现了),
我将它命名为jnitest,IDE自动生成C++的resource文件和header文件如图
其中jnitest.cpp添加代码如下代码如下,
#include "jnitest.h"
char *encode(char *data) {
return data;
}
jnitest.h添加代码如下
#ifndef MYJNIAPPLICATION_JNITEST_H
#define MYJNIAPPLICATION_JNITEST_H
#endif //MYJNIAPPLICATION_JNITEST_H
char *encode(char *data);
3.然后修改native-lib.cpp如下:
include “jnitest.h”
是native-lib.cpp中添加这个c++/c的依赖,相当于与java的导包,依赖的这个c++/c的头文件如图:
增加代码如下:
JNICALL JNIEXPORT jstring Java_com_example_gerryrun_myjniapplication_JniTest_Encode(JNIEnv *env,
jobject jobject1,
jstring parm1) {
char *target = (char *) env->GetStringUTFChars(parm1, 0);
char *result = encode(target);
return env->NewStringUTF(result);
}
4.调用
然后就是调用方法了,
// Example of a call to a native method
TextView tv = (TextView) findViewById(R.id.sample_text);
tv.setText(
JniTest.stringFromJNIString()
+ "/n"
+ JniTest.Encode("jni测试"));
clean项目然后Make project,run。在手机上就会看到实现的效果。
按上述步骤即可实现使用Android studio cMake打包.so库,.so的路径在build——>intermediates——>cmake下,因为在自己项目中做,每次编译都会进行cMake操作,耗时也没必要。所以建议新建一个和自己项目的包名一致的项目专门生成.so库的工作,把生成的.so库引入到自己项目中使用即可。
C++的小白编写[笑哭]