①:静态注册
在Java类中声明一个native方法,然后alt+enter,选择第一项,对应的dynamic.cpp文件中,就会生成对应的JNI方法。
注意看:JNI方法上方有JNIEXPORT和JNICALL两个宏定义声明,且其命名符合native方法和so方法的对应规则。其中JNIEnv类型代表了Java环境,通过这个JNIEnv* 指针,就可以对Java端的代码进行操作。jobject thiz代表该native方法的类实例或者这个native方法的类的class对象实例
优点:实现简单
缺点:
1、方法命名必须遵循Java_包名按下划线分拆_类名_方法名
2、JNI方法名过长
3、运行时根据函数名查找对应的JNI函数,程序效率不高;(查的,不太确定,有时间来验证一波)
②:动态注册
原理:在调用System.loadLibrary()时会在so层调用一个名为JNI_OnLoad()的函数,提供一个函数映射表,再在JNI_Onload()函数中通过JNI中提供的RegisterNatives()方法来注册函数。这样Java就可以通过函数映射表来调用函数,而不必通过函数名来查找对应函数。
dynamic.cpp代码:
/**
* 动态注册测试
*
* @author wangqikai5
* @version 1.0, 2022/1/19
* @since 产品模块版本
*/
#include <jni.h>
//NINativeMethod结构体,在jni.h中
//typedef struct {
// const char* name; // Java方法名
// const char* signature; // 方法签名描述
// void* fnPtr; // 函数指针,指向native函数
//} JNINativeMethod;
//动态注册测试方法
jint getNum(JNIEnv *env, jobject thiz, jint x){
return x*x;
}
//1.建立 函数映射表(本质为JNINativeMethod结构体数组)
JNINativeMethod methods[] = {
{"getNum", "(I)I", (void*)getNum}
};
//2.实现JNI_OnLoad方法,在加载动态库后,自动执行动态注册
jint JNI_OnLoad(JavaVM* vm, void* reserved){
JNIEnv* env=NULL;
if (vm->GetEnv((void**)&env,JNI_VERSION_1_6) != JNI_OK){
return JNI_ERR;
}
//3.调用FindClass方法,获取目标java对象
jclass clazz=env->FindClass("com/helper/jnidemo/MainActivity");
if (clazz == NULL){
return JNI_ERR;
}
//4.调用RegisterNatives方法,传入java对象、JNINativeMethod数组、注册数目,执行动态注册
jint result=env->RegisterNatives(clazz, methods, 1);
if (result < 0){ // 注册失败会返回一个负值
return JNI_ERR;
}
return JNI_VERSION_1_6;
}
//静态注册例子
extern "C"
JNIEXPORT jint JNICALL
Java_com_helper_jnidemo_MainActivity_dymaincTest(JNIEnv *env, jobject thiz) {
// TODO: implement dymaincTest()
}
Java类代码(MainActivity):
public class MainActivity extends AppCompatActivity {
/**
* 加载native库
*/
static {
System.loadLibrary("native-lib");
System.loadLibrary("Hero");
System.loadLibrary("dynamic");
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
TextView tv = findViewById(R.id.sample_text);
// tv.setText(JSON.toJSONString(getFlash()));
//调用动态注册方法,并展示结果
tv.setText(String.valueOf(getNum(6)));
}
public native String stringFromJNI();
/**
* 声明native方法
* @return
*/
public native Flash getFlash();
public native void test();
//静态注册例子
public native int dymaincTest();
//动态注册例子
public native int getNum(int x);
}
动态注册方法结果展示:
优点:
1、通过函数映射表来查找对应的JNI方法,运行效率高;(查的,不太确定,有时间来验证一波)
2、cpp内的JNI方法名不需要遵循静态注册那套命名规则,代码简洁;
缺点:实现步骤略微复杂;