- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在尝试为 Android 编写 MINPACK-Fortran 端口,但我无法理解在 JNI 中正确使用双数组。研究论坛上的问题,我意识到 JNI 在处理数组时需要大量无用的代码。我想是错误的,所以把代码带到这里。
在这个例子中,调用了过程hybrd1_(Fortran sub,由 MINPACK 提供)来计算非线性方程组。用户(Java 端)提供类 aSolver,其中包含方法 solveStep。 solveStep 有两个参数( double 组),其中 x - 参数 vector ; f - NLES 中右侧的 vector 。问题是在迭代继续时,我不知道 hybrd1_ 过程将哪些 vector 传输到 solveStep 中。但我知道,这些 vector 可以在 x、fvec 或 wa(aSolver 对象内的字段)中找到。hybrd1_ 调用 rhs 过程 (cdecl) - 我需要做很多可怕的事情才能将两个数组传输到 Java,然后将它们带回 c++。如您所见,rhs 过程调用非常频繁,这些 JNI 转换正在分解整个过程。
C++ 端
//struct which allow me to talk with Java from rhs procedure
struct fake_n
{
int n;
jobject aSolver;
jmethodID meth;
JNIEnv* env;
};
//iterator. the place where i need help
void rhs(int * n, double * x, double * f, int * flag)
{
fake_n* f_n = (fake_n *) n;
// creating array parametres
jdoubleArray x_row = ((*f_n).env)->NewDoubleArray((*f_n).n);
jdoubleArray f_row = ((*f_n).env)->NewDoubleArray((*f_n).n);
// filling array parametres
((*f_n).env)->SetDoubleArrayRegion(x_row, 0, (*f_n).n, x);
// launch object method
(*f_n).env->CallVoidMethod((*f_n).aSolver, (*f_n).meth, x_row, f_row);
// copy result back to c++ - why do i need?
jdouble *res = ((*f_n).env)->GetDoubleArrayElements(f_row, NULL);
for (int i =0; i< (*f_n).n; i++) { f[i] = res[i]; }
}
JNIEXPORT void JNICALL Java_com_example_myapplication_MainActivity_hybrd_1NLES(JNIEnv* env,
jclass clazz,
jobject aSolver)
{
jclass cls = env->GetObjectClass(aSolver);
jmethodID solvestep = env->GetMethodID(cls, "solveStep", "([D[D)V");
if (solvestep == 0) {
return;
}
jfieldID afields[4];
char* atype[4] = {(char *) "[I", (char *)"[D", (char *)"[D", (char *)"[D"};
char* aname[4] = {(char *)"n", (char *)"x", (char *)"fvec", (char *)"wa"};
jobject objs[4];
for (int i = 0; i < 4; i++) {
afields[i] = env->GetFieldID(cls, aname[i], atype[i]);
objs[i] = env->GetObjectField(aSolver, afields[i]);
}
jintArray *na = reinterpret_cast<jintArray *>(&(objs[0]));
jdoubleArray *xa = reinterpret_cast<jdoubleArray *>(&(objs[1]));
jdoubleArray *fveca = reinterpret_cast<jdoubleArray *>(&(objs[2]));
jdoubleArray *waa = reinterpret_cast<jdoubleArray *>(&(objs[3]));
int *n = env->GetIntArrayElements(*na, NULL);
double *x = env->GetDoubleArrayElements(*xa, NULL);
double *fvec = env->GetDoubleArrayElements(*fveca, NULL);
double *wa = env->GetDoubleArrayElements(*waa, NULL);
fake_n f_n;
f_n.n = *n;
f_n.aSolver = aSolver;
f_n.meth = solvestep;
f_n.env = env;
// tol, lwa, info - local variables
// this procedure calls the iterator (rhs) a lot of times
hybrd1_(rhs, (int *)&f_n, x, fvec, &tol, &info, wa, &lwa);
// should i ?
env->ReleaseIntArrayElements(*na, n, 0);
env->ReleaseDoubleArrayElements(*xa, x, 0);
env->ReleaseDoubleArrayElements(*fveca, fvec, 0);
env->ReleaseDoubleArrayElements(*waa, wa, 0);
}
这里是 JAVA 端
public class aSolver {
public double[] x;
public double[] fvec;
public double[] wa;
public int[] n;
public int rc;
public aSolver() {
n = new int[1];
n[0] = 2;
x = new double[n[0]];
fvec = new double[n[0]];
wa = new double[n[0] * (3 * n[0] + 13) / 2];
x[0] = 0;
x[1] = 1;
rc = 0;
solveStep(x, fvec);
}
// method, which calling from iterator (rhs)
public void solveStep(double[] xv, double[] fv ) {
// it is important: fv != fvec
// it is important: xv != x
fv[0] = 2.0 * xv[0] + 3.0 * xv[1] + 6.0;
fv[1] = 5.0 * xv[0] - 3.0 * xv[1] - 27.0;
rc++;
}
}
//----
aSolver s = new aSolver();
hybrd_NLES(s);
//----
public native void hybrd_NLES(aSolver solver);
书面代码看起来很繁琐且效率低下。此外,感觉 rhs sub 需要在 ReleaseDoubleArrayElements 中,但我不明白为什么需要那里。请帮我写一个高效的包装器。
最佳答案
首先,JNI 代码的定义确实很麻烦,因为您必须使用“env”来创建某些变量,并且您必须进行一些编码以使用 Java 代码获取所需的数据并与它。我真的不能说如何让你的代码更有效率,因为我真的不明白你想做什么。你可能有一些不必要的复制,我不确定。如果你检查你的代码并且仍然认为你可以改进它,请进一步解释你的目标是什么。关于“ReleaseDoubleArrayElements”的问题 - NewDoubleArray 仅创建本地引用,因此您无需发布它。
关于java - 使用双数组的最佳实践,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42167179/
我正在尝试使用内联汇编进行试验,我正在尝试在内联汇编中添加十进制数(不,不是整数)。问题是,当我调用以下函数时: inline double ADD(double num1, double num2)
我有一个名为“a”的数据类型为 double 的数字,当这个数字小于 0.5 时,说b = 1 - a没有错误。但是当 a > 0.5 时,我说b = 1 - a代码中的该点存在错误(执行该特定行之前
我需要将双数添加到 ol 列表中。一个例子是: 欢迎访客1.1.介绍1.2.更多信息1.3.更多信息1.4.更多信息1.5.更多信息 另一个OL2.1.更多信息2.2.更多信息2.3.更多信息 我知道
我正在尝试将一些代码从 C++ 转换为 JavaScript,但我对 int、doubles、NaN 和解析 float 感到非常困惑。我想!? Z = T - 14 - (1); while (Z
我是一名优秀的程序员,十分优秀!