Java JNI返回C++中结构

本文探讨如何在C++中使用JNI调用返回结构数组指针的函数,并将C++的结构转换为Java对象。问题涉及如何在Java中表示和处理C++的_STRUCTREC类型的数据,包括可能需要创建Java类数组并映射字段。

我想在C ++中调用一个返回结构数组指针的函数,像这样
BOOL __stdcall GetPositionBlotter(int* iCount, void* RecArray);

Where RecArray is elements like this

typedef struct _POSITIONREC {
char SymIdent[20];
char SymExchange[15];
char SymSecType[20];
char SymExpiry[30];
char SymRight[30];
char SymCurrency[20];
DWORD Quantity;
double MarketPrice;
double MarketValue;
double AverageCost;
double UnrealizedPNL;
double RealizedPNL;
} POSITIONREC, *LPPOSITIONREC;
;

如何将C ++结构恢复为Java? 需要一个类的数组吗? 需要以某种方式在C ++中实例化这些类,并在循环中加载字段并将它们作为结果传递回去。

非常简单但完整的工作示例。 仔细研究:

// 
// Main.java
// 
public class Main {
    static {
        System.loadLibrary("Main");
    }
     
    public static void main(String[] args) {
        new Main().exec();
    }
     
    void exec() {
        PositionRec[] posRecArray = getPositionBlotter();
         
        if (posRecArray != null) {
            for (int i = 0; i < posRecArray.length; i++)
                System.out.println(posRecArray[i]);
        }
    }
     
    native PositionRec[] getPositionBlotter();
}
  
// 
// PositionRec.java
// 
public class PositionRec {
    public String symIdent;
    public int quantity;
    public double marketPrice;
    public String toString() {
        return "[" + symIdent + "," + quantity + "," + marketPrice + "]";
    }
}
  
// 
// Main.h
// 
#ifndef _Included_Main
#define _Included_Main
  
#include <jni.h>
  
#ifdef __cplusplus
extern "C" {
#endif
  
/*
 * Class:     Main
 * Method:    getPositionBlotter
 * Signature: ()[LPositionRec;
 */
JNIEXPORT jobjectArray JNICALL Java_Main_getPositionBlotter(JNIEnv * env, jobject obj);
  
#ifdef __cplusplus
}
#endif
  
#endif /* _Included_Main */
  
// 
// Main.cpp
// 
#include <windows.h>
#include "Main.h"
  
typedef struct _POSITIONREC {
    char SymIdent[20];
    DWORD Quantity;
    double MarketPrice;
} POSITIONREC, * LPPOSITIONREC;
  
BOOL __stdcall GetPositionBlotter(int * iCount, void * recArray) {
    LPPOSITIONREC pPosRec = new POSITIONREC[2];
    strcpy(pPosRec[0].SymIdent, "Hello");
    pPosRec[0].Quantity = 1;
    pPosRec[0].MarketPrice = 2.0;
    strcpy(pPosRec[1].SymIdent, "World!");
    pPosRec[1].Quantity = 3;
    pPosRec[1].MarketPrice = 4.0;
    *iCount = 2;
    *((LPPOSITIONREC *)recArray) = pPosRec;
    return TRUE;
}
  
typedef struct _JNI_POSREC {
    jclass cls;
    jmethodID ctorID;
    jfieldID symIdentID;
    jfieldID quantityID;
    jfieldID marketPriceID;
} JNI_POSREC;
  
JNI_POSREC * jniPosRec = NULL;
  
void loadJniPosRec(JNIEnv * env) {
    if (jniPosRec != NULL)
        return;
    jniPosRec = new JNI_POSREC;
    jniPosRec->cls = env->FindClass("PositionRec");
    jniPosRec->ctorID = env->GetMethodID(jniPosRec->cls, "<init>", "()V");
    jniPosRec->symIdentID = env->GetFieldID(jniPosRec->cls, "symIdent", "Ljava/lang/String;");
    jniPosRec->quantityID = env->GetFieldID(jniPosRec->cls, "quantity", "I");
    jniPosRec->marketPriceID = env->GetFieldID(jniPosRec->cls, "marketPrice", "D");
}
  
void fillJavaPosRecValues(JNIEnv * env, jobject jPosRec, POSITIONREC cPosRec) {
    env->SetObjectField(jPosRec, jniPosRec->symIdentID, env->NewStringUTF(cPosRec.SymIdent));
    env->SetIntField(jPosRec, jniPosRec->quantityID, cPosRec.Quantity);
    env->SetDoubleField(jPosRec, jniPosRec->marketPriceID, cPosRec.MarketPrice);
}
  
/*
 * Class:     Main
 * Method:    getPositionBlotter
 * Signature: ()[LPositionRec;
 */
JNIEXPORT jobjectArray JNICALL Java_Main_getPositionBlotter(JNIEnv * env, jobject obj) {
    loadJniPosRec(env);
     
    LPPOSITIONREC pPosRec;
    int count;
     
    if (!GetPositionBlotter(&count, (void *)&pPosRec))
        return NULL;
     
    jobjectArray jPosRecArray = env->NewObjectArray(count, jniPosRec->cls, NULL);
     
    for (int i = 0; i < count; i++) {
        jobject jPosRec = env->NewObject(jniPosRec->cls, jniPosRec->ctorID);
         
        fillJavaPosRecValues(env, jPosRec, pPosRec[i]);
        env->SetObjectArrayElement(jPosRecArray, i, jPosRec);
    }
     
    return jPosRecArray;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值