27

我需要通过 Java

List< List<MyPoint> > points;

通过 jni 到 C++ 并转换为

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

处理这个向量并返回

List< List<MyPoint> >
  1. 如何正确传递列表和返回列表?
  2. 如何在对象向量的向量中和向后转换对象列表列表?
4

4 回答 4

18

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

  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;

Java部分

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

于 2012-06-07T05:02:37.973 回答
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;    

}
于 2014-05-01T03:37:39.637 回答
2

正如我从参考 JNI 中理解的那样,JNI 只能处理原始类型或对象的一维数组。

因为在Java方面,不得不将列表翻译成数组。然后,在本机部分传递的数组和元素的数量。将转到所需的向量并进行处理。返回两个数组(包含所有轮廓点的数组和包含每个轮廓中的点数的数组)和轮廓数的结果。结果数组收集在 Java 一侧的列表列表中。

虽然问题没有完全解决,因为 JNI 无法为 native 部分中的现有项分配内存。因此需要将部分数据提取出来,在Java端为它们分配内存,并填入native。

一个可能的解决方案可能是使用 SWIG 或 JavaCpp 之类的活页夹

于 2012-06-02T10:02:48.933 回答
1

你也可以使用这个项目。它将允许像使用原生类一样使用 JNI 下的 java 类。

于 2012-08-05T00:40:06.440 回答