使用 Android NDK 获取 ANDROID_ID - 过时的本地引用错误

Get ANDROID_ID using Android NDK - stale local reference error

我正在开发一个应用程序,其中一个要求是我通过 NDK 收集设备的 ANDROID_ID

在查看了无数的线程、答案和参考资料之后,我想出了一个初始实现,它在 class 上调用一个静态方法,我用来在 Java 和本机层之间进行所有交互.唯一的问题是它在检索 ANDROID_ID 时在底部抛出以下错误,然后就退出了。

JNI ERROR (app bug): accessed stale local reference 0x5d6892a9 (index 9386 in a table of size 11)

我知道错误发生在最后一个函数调用 (CallStaticObjectMethod) 上,因为我已经尝试向它添加日志并且它会执行其他所有操作。这是我目前使用的代码:

SampleActivity.java

class SampleActivity {

    // ...

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        NativeService.initialize(this);

        // ...
    }
}

NativeService.java

public class NativeService {

    // ...

    public native static boolean initialize(Context ctx);
}

Native.cpp

JNIEXPORT jboolean JNICALL Java_com_company_mobile_NativeService_initialize(JNIEnv * env, jobject obj,
    jobject ctx
) {
    jclass contextClass = env->GetObjectClass(ctx);
    if (contextClass == NULL) {
        return false;
    }

    jmethodID getContentResolverMID = env->GetMethodID(contextClass, "getContentResolver", "()Landroid/content/ContentResolver;");
    if (getContentResolverMID == NULL) {
        return false;
    }

    jobject contentResolverObj = env->CallObjectMethod(ctx, getContentResolverMID);
    if (contentResolverObj == NULL) {
        return false;
    }

    jclass settingsSecureClass = env->FindClass("android/provider/Settings$Secure");
    if (settingsSecureClass == NULL) {
        return false;
    }

    jmethodID getStringMID = env->GetStaticMethodID(settingsSecureClass, "getString", "(Landroid/content/ContentResolver;Ljava/lang/String;)Ljava/lang/String;");
    if (getStringMID == NULL) {
        return false;
    }

    // Offending line
    jstring androidId = (jstring) env->CallStaticObjectMethod(settingsSecureClass, getStringMID, contentResolverObj, "android_id");
    if (androidId == NULL) {
        return false;
    }

    return (strcmp((char *)androidId, "0123456789ABCDEF") == 0);
}

getString 的第二个参数应该是 java/lang/String 的实例。文字 "android_id" 是一个 char 的数组,它会衰减为一个 (const) char*.

要从 const char* 字符串构造一个 java/lang/String 实例,您应该使用 JNI 函数 NewStringUTF:

jstring idStr = (jstring) env->NewStringUTF("android_id");
// Do relevant error checking, and then:
jstring androidId = (jstring) env->CallStaticObjectMethod(settingsSecureClass, getStringMID, contentResolverObj, idStr);