首页 > 解决方案 > 具有原始类型的 JNI GetObjectClass

问题描述

我有以下 JNI/JVMTI 代码:

jfieldID field = ...;
jobject fieldValue = (*jni_env)->GetObjectField(jni_env, jObjectInstance, field);
jclass fieldClass = (*jni_env)->GetObjectClass(jni_env, fieldValue);
char* signature_ptr;
char* generic_ptr;
(*jvmti_env)->GetClassSignature(jvmti_env, fieldClass, &signature_ptr, &generic_ptr);

此代码适用于 long字段是引用类型(java.lang.Object 和子类型)。如果field是原始类型,它会在尝试执行GetObjectClass时使 JVM 崩溃。

我的问题是:如何使用fieldfieldValue来确定它是否是原始类型以及是否是原始类型,是哪一个(int、long、boolean 等)?

我是 JNI/JVMTI 的新手,所以如果答案很明显,请多多包涵。

谢谢!

标签: javajvmjava-native-interfacejvmti

解决方案


调用which 不是对象类型是非法GetObjectField的。jfieldID对于原始字段,您需要调用其他Get[Type]Field函数之一,其中[Type]Int// Long/ Booleanetc。

既然你已经有了jvmti_env,查找jfieldID类型的最简单方法是调用 JVM TI GetFieldName函数。除其他外,它返回字段签名,即字段的类型。

唯一的问题是jfieldID应该始终伴随jclass- 字段的持有者,因为字段 ID 是相对于特定类的。通常,当您获得 时jfieldID,您已经知道它的持有者类。如果没有,在您的情况下,您有jObjectInstance- 一个具有此字段的对象,因此可以使用 JNI GetObjectClass函数从实例派生持有者类。

jclass fieldHolder = (*jni_env)->GetObjectClass(jni_env, jObjectInstance);

char* name;
char* signature;
if ((*jvmti_env)->GetFieldName(jvmti_env, fieldHolder, field, &name, &signature, NULL) == 0) {

    switch (signature[0]) {
        case 'I':
            // int
            break;
        case 'J':
            // long
            break;
        ...
        case 'L':
            // object
            break;
        case '[':
            // array
            break;
    }

    (*jvmti_env)->Deallocate(jvmti_env, signature);
    (*jvmti_env)->Deallocate(jvmti_env, name);
}

推荐阅读