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

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

和-



但是此错误很少会影响javahjavac -h dir生成的 header ,因为通常本机方法是使用“短”名称生成的,例如不在乎参数类型的Java_A_a

解决方案是按照https://bugs.openjdk.java.net/browse/JDK-8145897的建议手动更改方法签名。

10-08 18:18