Android JNI 静态与动态注册

①:静态注册

在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方法名不需要遵循静态注册那套命名规则,代码简洁;

缺点:实现步骤略微复杂;

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值