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

Ant*_*ony 4 c++ java java-native-interface android javah

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

Landroid_graphics_Bitmap_Config
Run Code Online (Sandbox Code Playgroud)

这相当于:

Landroid/graphics/Bitmap/Config
Run Code Online (Sandbox Code Playgroud)

代替:

Landroid_graphics_Bitmap_00024Config
Run Code Online (Sandbox Code Playgroud)

这是等效的

Landroid/graphics/Bitmap$Config
Run Code Online (Sandbox Code Playgroud)

javah 生成的结果是错误的,因为 JNI 会在寻找内部类的_00024表示时抛出错误$。man forjavah似乎没有暗示任何设置来纠正这个问题。这只是限制javah吗?

Ale*_*ohn 5

看起来,当涉及内部类类型的参数时,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);
    }
}
Run Code Online (Sandbox Code Playgroud)

如果使用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
Run Code Online (Sandbox Code Playgroud)

和 -

java.lang.UnsatisfiedLinkError:没有找到 void Aa(android.graphics.Bitmap$Config) 的实现(尝试了 Java_A_a 和 Java_A_a__Landroid_graphics_Bitmap_00024Config_2)

但是这个错误很少影响由javahor生成的头文件javac -h dir,因为通常本地方法是用“短”名称生成的,例如Java_A_a它不关心参数的类型。

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