27

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

List< List<MyPoint> > points;

jni 経由で C++ に変換し、

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

このベクトルを処理して返す

List< List<MyPoint> >
  1. リストのリストをどのように渡し、返すのが正しいですか?
  2. オブジェクトのベクトルのベクトルと逆方向のオブジェクトのリストのリストをどのように変換しますか?
4

4 に答える 4

18

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

  1. オブジェクト (O) コンテナー (C) として Java クラスに作成します。
  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はプリミティブ型またはオブジェクトの1次元配列でのみ機能します。

Java 側では、リストを配列に変換する必要があったためです。次に、ネイティブ部分で、渡された配列と要素の数。目的のベクトルに行き、処理されます。2 つの配列 (すべてのコンターのポイントを持つ配列と各コンターのポイント数を持つ配列) とコンターの数の結果として返されます。結果の配列は、Java 側のリストのリストに収集されます。

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

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

于 2012-06-02T10:02:48.933 に答える
1

このプロジェクトを使用することもできます。ネイティブのような JNI を使用する Java クラスを使用できるようになります。

于 2012-08-05T00:40:06.440 に答える