Android底层开发之动态注册和附加线程

学习Android NDK开发,了解JNI层的反射之后,就要开始逼格更高的知识学习了。今天我们讲两个东西,方法的动态注册和附加Native线程到JVM。

方法的静态注册与动态注册

静态注册

什么是方法的静态注册?即我们硬编码,按照一定的规则,映射Java本地方法和Native函数,使其自动关联绑定的过程。静态注册是不需要我们写额外的代码的,它的映射规则很简单,即Java_包名(.换_,_换_1),然后参数和返回值跟JNI类型对应即可,注意第一个函数参数一定是JNIEnv *env,第二个函数参数一定是jobject thiz,如果是静态方法,则可以替换为jclass clazz,jobject是兼容jclass的。

动态注册

说起动态注册,咱们不得不聊聊JNI_OnLoad函数。

cpp 复制代码
jint JNI_OnLoad(JavaVM *vm, void *reserved) {
    return JNI_VERSION_1_6;
}

每一个动态库so文件中只能定义一个名为JNI_OnLoad的函数,就跟定义main函数一样。在Java代码中调用System.loadLibary("库名称");的时候,就会调用JNI_OnLoad函数。在这里我们可以做一些库的初始化操作。通常我们在这里就可以拿到JNIEnv的指针,而无需通过硬编码native函数传入。

cpp 复制代码
jint JNI_OnLoad(JavaVM *vm, void *reserved) {
    JNIEnv *env = NULL;
    // 获取JNIEnv
    if (vm->GetEnv((void **) &env, JNI_VERSION_1_6) != JNI_OK) {
        return -1;
    }
    // 在此做你的事情
    return JNI_VERSION_1_6;
}

在这里我们就可以去反射Java代码了。那么重点来了,如何进行方法的动态注册?

cpp 复制代码
JNIEXPORT jstring JNICALL
yourMethod(JNIEnv *env, jobject thiz, jstring jstr) {
    return (jstr == nullptr) ? nullptr : (jstring) env->NewLocalRef(jstr);
}

// 构建 JNINativeMethod 的数组
static JNINativeMethod methods[] = {
        {"yourMethod", "(Ljava/lang/String;)Ljava/lang/String;", (void *) yourMethod},
};

jint JNI_OnLoad(JavaVM *vm, void *reserved) {
    JNIEnv *env = NULL;
    // 获取JNIEnv
    if (vm->GetEnv((void **) &env, JNI_VERSION_1_6) != JNI_OK) {
        return -1;
    }
    // 获取需要注册的方法所在的Java类
    jclass clazz = env->FindClass("com/example/MyClass");
    if (clazz == NULL) {
        return -1;
    }
    // 进行方法的动态注册
    if (env->RegisterNatives(clazz, methods, sizeof(methods) / sizeof(methods[0])) < 0) {
        // 注册失败
        return -1;
    }
    return JNI_VERSION_1_6;
}

我们总结下使用动态注册的好处和什么情况下需要使用方法的动态注册。

  • 需要提高安全性,防止被 Java 反射直接调用

  • Java 类名可能会变,不希望 C/C++ 代码跟着改

  • 需要在不同环境下运行时修改或替换 JNI 方法

附加线程与分离线程

我们可以通过调用javaVM->AttachCurrentThread(&jniEnv, nullptr);将当前线程附加到JVM。由于JNIEnv指针不能跨线程使用,需要先附加线程到JVM,才能调用JNI方法。例如:

cpp 复制代码
JavaVM *g_JavaVM;

JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved) {
    g_JavaVM = vm;  // 保存 JavaVM 实例
    return JNI_VERSION_1_6;
}

void *thread_function(void *arg) {
    JNIEnv *env;
    // 附加当前线程到 JVM
    if (g_JavaVM->AttachCurrentThread(&env, nullptr) != JNI_OK) {
        return nullptr;
    }
    // 现在可以安全调用 Java 方法
    jclass clazz = env->FindClass("com/example/MyClass");
    jmethodID method = env->GetStaticMethodID(clazz, "callback", "()V");
    env->CallStaticVoidMethod(clazz, method);
    // 线程执行完成后,分离当前线程
    g_JavaVM->DetachCurrentThread();
    return nullptr;
}

最后我们需要使用POSIX创建一个可移植操作系统的线程,使用的是pthread_t

cpp 复制代码
pthread_t thread;
pthread_create(&thread, nullptr, thread_function, nullptr);

总结

那么,本篇的内容就讲完了,主要讲解了JNI入口函数以及如何进行方法的动态注册,最后讲解了JNIEnv不能跨线程使用,所以需要附加线程到JVM。本篇是NDK底层开发的基础,非常重要,JYM务必重视起来,慢慢消化掌握。

相关推荐
有位神秘人43 分钟前
Android中获取当前屏幕的宽高工具类
android
Yang-Never1 小时前
Open GL ES -> 应用前后台、Recent切换,SurfaceView纹理贴图闪烁问题分析解决
android·开发语言·kotlin·android studio·贴图
liujun35121591 小时前
camera开发,我对预览请求的理解
android
无法长大2 小时前
Mac M1 环境下使用 Rust Tauri 将 Vue3 项目打包成 APK 完整指南
android·前端·macos·rust·vue3·tauri·打包apk
一只程序熊2 小时前
uniappx 配置 uni.chooseLocation 地图
android·uni-app x
Yang-Never2 小时前
Android 应用启动 -> Android 多种方式启动同一进程,Application.onCreate() 会多次执行吗?
android·java·开发语言·kotlin·android studio
2501_916008892 小时前
iOS 开发助手工具,设备信息查看、运行日志、文件管理等方面
android·ios·小程序·https·uni-app·iphone·webview
TheNextByte12 小时前
如何在Android上恢复已删除的文件
android·gitee
梁同学与Android2 小时前
Android ---【Kotlin篇】Kotlin 协程中 StateFlow 与 SharedFlow 的网络状态对比与应用
android·网络·kotlin
Fate_I_C2 小时前
Android Jetpack实战
android·android jetpack