我有一个 JNI 函数,它android.graphics.Bitmap$Config
作为参数传递。 Config
是Bitmap
的内部类。 当我运行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(
但是这个错误很少影响javah
或javac -h dir
生成的标头,因为通常本机方法是用"短"名称生成的,例如 Java_A_a
不关心参数的类型。
解决方案是手动更改方法签名,如 https://bugs.openjdk.java.net/browse/JDK-8145897 中所述。