web-dev-qa-db-ja.com

JNIを介してリストのリストを渡し、返し、ベクトルに変換します

Javaから渡す必要があります

List< List<MyPoint> > points;

jniを介してC++に変換し、

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

このベクトルを処理して、

List< List<MyPoint> >
  1. リストのパスとリターンのリストはどの程度正しいですか?
  2. オブジェクトのベクトルのベクトル内のオブジェクトのリストのリストをどのように逆方向に変換しますか?
26
George

私はこの問題を標準的なツールで解決しました。

  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-返された配列から行のリストを作成します

16
George
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;    

}
4
user2264907

JNIのリファレンスから理解しているように、JNIはプリミティブ型またはオブジェクトの1次元配列でのみ機能します。

Java側では、リストを配列に変換する必要があったためです。次に、ネイティブ部分で渡された配列と要素の数。目的のベクトルに移動して処理されます。 2つの配列(すべての等高線の点を含む配列と各等高線の点の数を含む配列)と等高線の数の結果として返されます。結果の配列は、Java側のリストのリストに収集されます。

問題は完全には解決されていませんが、JNIはネイティブ部分の既存のアイテムにメモリを割り当てることができないためです。したがって、データを部分的に抽出し、Java側でそれらにメモリを割り当て、ネイティブに入力する必要があります。

考えられる解決策は、SWIGやJavaCppなどのバインダーの使用です。

2
George

これも使用できます project 。 Javaクラスは、ネイティブクラスのようにJNIを使​​用できます。

1