Я пытаюсь вызвать функцию C ++ из Java через JNA. Я хочу передать строку и получить строку обратно. Это делается с помощью входного параметра и выходного параметра. я использую PointerByReference
представлять char**
выходной параметр. Вызов C ++ работает, но PointerByReference
является нулевым после вызова.
Я основал свой код на PointerByReference Docs. Есть идеи, что я делаю не так?
Я добавил операторы печати в C ++, чтобы убедиться, что он не устанавливает указатель в null, и это определенно не так. Поэтому некоторые в моем использовании JNA, должно быть, ошибаются. Но что?
Код C ++
void processRequest(char* input, char** output, int* outputLength)
{
// Variable output size from processInput
std::string sOutput = processInput(input);char* results = new char[sOutput.length() + 1];
strncpy(results, sOutput.c_str(), sOutput.length());
results[sOutput.length()] = '\0';output = &results;
outputLength = new int(strlen(results) + 1);
}
Java-код
Который называется Java следующим образом:
public class RunRequestT1 {
private static final Logger log = LoggerFactory.getLogger(MyLibWrapperImpl.class);
static boolean envVarsSetupDone = false;
static interface MyLib extends Library {
MyLib INSTANCE = (MyLib) Native.loadLibrary("N:\\sys1\\sys1_dist\\MyLib\\MyLib9.8.Q.P3.G15.T\\bin\\MyLib.dll", MyLib.class);
public void processRequest(String request, PointerByReference bufp, IntByReference lenp);
public void clearMemoryPtr(PointerByReference bufp, IntByReference lenp);
}
public static void main(String[] args) throws IOException {
System.out.println("sys1: Reading request");
String request = FileUtils.readFileToString(new File("C:\\dev\\prj\\single-request.xml"), "UTF-8");
System.out.println("sys1: Pricing request");
String response = processRequest(request);
System.out.println(response);
}
public static String processRequest(String request) {
System.out.println("sys1: prepare args");
// code based on https://github.com/twall/jna/blob/master/www/ByRefArguments.md
PointerByReference bufp = new PointerByReference();
IntByReference lenp = new IntByReference();
System.out.println("sys1: making call");
MyLib.INSTANCE.processRequest(request, bufp, lenp);
System.out.println("reading response");
System.out.println("bufp: " + bufp.getValue());
System.out.println("lenp: " + lenp.getValue());
Pointer p = bufp.getValue();
byte[] buffer = p.getByteArray(0, lenp.getValue());
String response = Native.toString(buffer);
//String response = p.getString(0);
// de-allocate memory buffer
System.out.println("cleaning memory");
MyLib.INSTANCE.clearMemoryPtr(bufp, lenp);
return response;
}
}
Stdout
И когда я это называю, я получаю такие результаты:
sys1: Reading request
sys1: Pricing request
sys1: prepare args
sys1: making call
reading response
bufp: null
lenp: 0
Exception in thread "main" java.lang.NullPointerException
at com.calyon.gcm.MyLibwrapper.main.RunRequestT1.priceRequest()
at com.calyon.gcm.MyLibwrapper.main.RunRequestT1.main()
Я пытался сделать его минимальным, и, действительно, ничего особенного не происходит.
Я тестировал с JNA версий 3.4 и 3.5
Вместо того, чтобы выделять новый указатель для вашей длины «return», вам нужно записать в существующий указатель, например,
*outputLength = strlen(results) + 1;
Вам также нужно записать в заданный указатель буфера, а не просто присвоить новое значение вашему локальному аргументу:
*output = results;
Других решений пока нет …