通过 JNI 传递、返回和转换为向量列表列表

Pass, return and convert to vectors list of lists over JNI

本文关键字:列表 转换 向量 返回 JNI 传递 通过      更新时间:2023-10-16

我需要从Java传递

List< List<MyPoint> > points;

通过 JNI C++并转换为

std::vector< std::vector<MyPoint> >

处理此向量并返回

List< List<MyPoint> >
  1. 如何正确通过和返回列表列表?
  2. 如何在对象向量向量中转换对象列表列表并向后转换?

我用标准工具解决了这个问题。

  1. 在 Java 类中创建为对象 (O) 容器 (C)
  2. 将对象数组 (O) 从 Java 代码传递到本机部分
  3. 从C++代码中的数组向量创建
  4. 计算新向量
  5. 构建容器数组 (C) 并插入到对象 (O)
  6. 返回容器数组 (C)

代码实现:

在Java部分:

1 - 从点列表创建数组

在 c++ 部分:

2 - 构建输入向量

std::vector<CurvePoint> src_line;
jclass java_points_cls = env->FindClass("myPointClass");
jmethodID java_mid = env->GetMethodID(java_points_cls, "<init>", "(II)V");    
jfieldID fidX = env->GetFieldID(java_points_cls, "x", "I");
jfieldID fidY = env->GetFieldID(java_points_cls, "y", "I");
int srcCount = env->GetArrayLength(srcLines);
for (int i=0; i < srcCount; i++) 
{
    jobject cur_pnt =  env->GetObjectArrayElement(srcLines, i); 
    LinePoint src_point;        
    src_point.x = env->GetIntField(cur_pnt, fidX); 
    src_point.y = env->GetIntField(cur_pnt, fidY);    
    src_line.push_back(src_point);
}

3 - 计算lines

4 - 构建输出数组

jclass java_line_cls = env->FindClass("myLinesClass");
jmethodID java_line_add = env->GetMethodID(java_line_cls, "addPoint", "(II)V");  
jmethodID java_line_init = env->GetMethodID(java_line_cls, "<init>", "()V");
jobjectArray resLines = (jobjectArray) env->NewObjectArray(lines.size(),     java_line_cls, 0); 
for(int i = 0; i < lines.size(); ++i)
{
    jobject cur_line =  env->NewObject(java_line_cls, java_line_init);
    for(int j = 0; j < lines[i].size(); ++j)
        env->CallVoidMethod(cur_line, java_line_add, 
                                lines[i][j].x,
                                lines[i][j].y);
    env->SetObjectArrayElement(resLines, i, cur_line);
}
return resLines;

爪哇部分

5 - 从返回的数组创建行列表

JNIEXPORT jobjectArray JNICALL Java_ProcessInformation_getAllProcessPid  (JNIEnv*env,jobject obj) {
    vector<string>vec;
    vec.push_back("Ranjan.B.M");
    vec.push_back("Mithun.V");
    vec.push_back("Preetham.S.N");
    vec.push_back("Karthik.S.G");
    cout<<vec[0];
    cout<<vec[0];
    jclass clazz = (env)->FindClass("java/lang/String");
    jobjectArray objarray = (env)->NewObjectArray(vec.size() ,clazz ,0);
    for(int i = 0; i < vec.size(); i++) {
        string s = vec[i]; 
         cout<<vec[i]<<endl;
         jstring js = (env)->NewStringUTF(s.c_str());
        (env)->SetObjectArrayElement(objarray , i , js);
    }
    return objarray;    
}

正如我从 JNI 参考资料中了解到的那样,JNI 只能处理基元类型或对象的一维数组。

因为在Java方面,不得不将列表转换为数组。然后,在本机部分中传递数组和元素数。有所需的矢量并处理。返回两个数组(包含所有轮廓的点的数组和每个轮廓中具有点数的数组)和等高线数的结果。生成的数组收集在 Java 方面的列表列表中。

虽然问题没有完全解决,因为 JNI 无法为本机部分中的现有项目分配内存。因此,有必要部分提取数据,在Java方面为它们分配内存,并填充本机数据。

一个可能的解决方法可能是使用粘合剂,如SWIG或JavaCpp

您也可以使用此项目。它将允许像本地类一样使用 java 类欠 JNI。