如何在 JNI 中访问 Pair<> of Java?
How to access Pair<> of Java in JNI?
我在 Java 端有 ArrayList<Pair<Pair<Float, Float>, Pair<Float, Float>>>
,想在 JNI 中使用这些数据。
用什么方法和类转换成std::pair<std::pair<float, float>, std::pair<float, float>>
我尝试关注
jclass pairClass = env->FindClass("android/util/Pair");
jfieldID pairGetKey = env->GetFieldID(pairClass, "first", "java/util/Objects");
pairGetKey
始终为空
字段类型必须作为签名给出;也就是说,您需要使用 int
-> I
、T[]
-> [T
、reference.Type
-> Lreference/Type;
编码。另外,字段的类型是java.lang.Object
,不是java.util.Objects
.
jfieldID first = env->GetFieldID(pairClass, "first", "Ljava/lang/Object;");
jfieldID second = env->GetFieldID(pairClass, "second", "Ljava/lang/Object;");
剩下的很乏味,但并不难:
jfloat extract_float(JNIEnv *env, jobject f) {
// Note the syntax of signatures: float floatValue() has signature "()F"
return env->CallFloatMethod(f,
env->GetMethodID(env->FindClass("java/lang/Float"), "floatValue", "()F"));
}
std::pair<jobject, jobject> extract_pair(JNIEnv *env, jobject p) {
jclass pairClass = env->FindClass("android/util/Pair");
jfieldID first = env->GetFieldID(pairClass, "first", "Ljava/lang/Object;");
jfieldID second = env->GetFieldID(pairClass, "second", "Ljava/lang/Object;");
return std::pair(env->GetObjectField(p, first), env->GetObjectField(p, second));
}
JNIEnv *env;
jobject pair;
auto [f1, f2] = extract_pair(env, pair);
auto [f11, f12] = extract_pair(env, f1);
auto [f21, f22] = extract_pair(env, f2);
std::pair p(
std::pair(extract_float(env, f11), extract_float(env, f12)),
std::pair(extract_float(env, f21), extract_float(env, f22)));
虽然,我想我必须问,你真的需要这样做吗?您可以将嵌套对预处理为 Java 端更好的东西吗?在这边做很丑
我在 Java 端有 ArrayList<Pair<Pair<Float, Float>, Pair<Float, Float>>>
,想在 JNI 中使用这些数据。
用什么方法和类转换成std::pair<std::pair<float, float>, std::pair<float, float>>
我尝试关注
jclass pairClass = env->FindClass("android/util/Pair");
jfieldID pairGetKey = env->GetFieldID(pairClass, "first", "java/util/Objects");
pairGetKey
始终为空
字段类型必须作为签名给出;也就是说,您需要使用 int
-> I
、T[]
-> [T
、reference.Type
-> Lreference/Type;
编码。另外,字段的类型是java.lang.Object
,不是java.util.Objects
.
jfieldID first = env->GetFieldID(pairClass, "first", "Ljava/lang/Object;");
jfieldID second = env->GetFieldID(pairClass, "second", "Ljava/lang/Object;");
剩下的很乏味,但并不难:
jfloat extract_float(JNIEnv *env, jobject f) {
// Note the syntax of signatures: float floatValue() has signature "()F"
return env->CallFloatMethod(f,
env->GetMethodID(env->FindClass("java/lang/Float"), "floatValue", "()F"));
}
std::pair<jobject, jobject> extract_pair(JNIEnv *env, jobject p) {
jclass pairClass = env->FindClass("android/util/Pair");
jfieldID first = env->GetFieldID(pairClass, "first", "Ljava/lang/Object;");
jfieldID second = env->GetFieldID(pairClass, "second", "Ljava/lang/Object;");
return std::pair(env->GetObjectField(p, first), env->GetObjectField(p, second));
}
JNIEnv *env;
jobject pair;
auto [f1, f2] = extract_pair(env, pair);
auto [f11, f12] = extract_pair(env, f1);
auto [f21, f22] = extract_pair(env, f2);
std::pair p(
std::pair(extract_float(env, f11), extract_float(env, f12)),
std::pair(extract_float(env, f21), extract_float(env, f22)));
虽然,我想我必须问,你真的需要这样做吗?您可以将嵌套对预处理为 Java 端更好的东西吗?在这边做很丑