JNI:javah 破坏了作为内部类的参数



我有一个 JNI 函数,它android.graphics.Bitmap$Config作为参数传递。 ConfigBitmap的内部类。 当我运行javah时,我得到不正确的标头签名(仅截断为单个参数(:

Landroid_graphics_Bitmap_Config

相当于:

Landroid/graphics/Bitmap/Config

而不是:

Landroid_graphics_Bitmap_00024Config

这是等效的

Landroid/graphics/Bitmap$Config

javah 生成的内容是错误的,因为 JNI 将抛出一个错误,为内部类查找$_00024表示形式。 javah的男人似乎没有暗示任何设置来纠正这一点。 这只是javah的限制吗?

看起来,当涉及内部类类型的参数时,JDK 中存在错误(或至少不一致(。

下面是重现该问题的示例类:

public class A {
    public native void a(android.graphics.Bitmap.Config b);
    public native void a(android.graphics.Bitmap.Config b, int c);
    static {
        System.loadLibrary("hello-libs");
        a(null);
    }
}

如果使用javah生成本机标头,您将获得

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_example_hellolibs_MainActivity */
#ifndef _Included_com_example_hellolibs_MainActivity
#define _Included_com_example_hellolibs_MainActivity
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     A
 * Method:    a
 * Signature: (Landroid/graphics/Bitmap/Config;)V
 */
JNIEXPORT void JNICALL A_a__Landroid_graphics_Bitmap_Config_2
  (JNIEnv *, jobject, jobject);
/*
 * Class:     A
 * Method:    a
 * Signature: (Landroid/graphics/Bitmap/Config;I)V
 */
JNIEXPORT void JNICALL Java_A_a__Landroid_graphics_Bitmap_Config_2I
  (JNIEnv *, jobject, jobject, jint);
#ifdef __cplusplus
}
#endif
#endif

和-

java.lang.UnsatisfiedLinkError: 找不到 void A.a(android.graphics.Bitmap$Config( 的实现(尝试Java_A_a和Java_A_a__Landroid_graphics_Bitmap_00024Config_2(

但是这个错误很少影响javahjavac -h dir生成的标头,因为通常本机方法是用"短"名称生成的,例如 Java_A_a不关心参数的类型。

解决方案是手动更改方法签名,如 https://bugs.openjdk.java.net/browse/JDK-8145897 中所述。

最新更新