我可以访问test.h和test.so文件:
test.h
int insert_data(void *location_data, uint64_t device_id, uint64_t event_id,
uint64_t motion_seq, uint64_t utc_time, uint32_t sensor_seq,
uint32_t device_seq, uint32_t tick_mark, uint16_t ble,
int movement, int path_id);
我现在正在编写JNA代码来调用上面的方法。
interface TestCLibrary extends Library {
TestCLibrary INSTANCE = (TestCLibrary) Native.loadLibrary(“test”, TestCLibrary);
int insert_data(Pointer test_data, long device_id, long event_id,
long motion_seq, long utc_time, int sensor_seq,
int device_seq, int tick_mark, short ble,
int movement, int path_id);
}
public class JNATest {
public static void main(String[] args) throws IOException{
Native.setProtected(true);
int path_id = 1, movement = 1, tick_mark = 1, device_seq = 1;
long utc_time = System.currentTimeMillis();
int sensor_seq = 100;
short ble = 25;
long motion_seq = 22222299;
long device_id = 111122333;
long event_id = 44445555;
Pointer test_data = Pointer.NULL;
TestCLibrary.INSTANCE.insert_data(test_data, device_id, event_id, motion_seq, utc_time, sensor_seq, device_seq, tick_mark, ble, movement, path_id);
}
}
在此代码位置运行JNATest时面临以下异常:
TestCLibrary.INSTANCE.insert_data
线程中的异常" main" java.lang.Error:无效的内存访问 com.sun.jna.Native.invokeInt(Native Method)at com.sun.jna.Function.invoke(Function.java:419)at com.sun.jna.Function.invoke(Function.java:354)at com.sun.jna.Library $ Handler.invoke(Library.java:244)at com.test.apps。$ Proxy0.insert_data(未知来源)....
如何避免此类错误? 以及如何使用默认值初始化JNA中的指针?
答案 0 :(得分:0)
错误正在发生,因为我假设该函数将数据写入传递给它的指针所指向的内存中。您需要分配函数可以写入的内存。它无法写入NULL
指针。
应该做什么取决于数据是什么,如何构建以及如何使用它。
如果数据只是一个整数或字节数组,请将其视为(相应地更改数组大小):
int insert_data(int[] data, // ... etc.
// ...
int[] data = new int[4096];
insert_data(data, /* ... */
但是,如果您知道该函数生成的数据是struct
或class
,请为其创建Java class
。 JNA将自动将由本机函数写入的数据“编组”到有意义的Java类实例中:
https://java-native-access.github.io/jna/4.2.1/com/sun/jna/Structure.html
https://github.com/java-native-access/jna/blob/master/www/StructuresAndUnions.md
由于函数接受指针,因此您需要使用Structure.ByReference
:
int insert_data(Structure.ByReference data, // ...
答案 1 :(得分:0)
执行以下步骤后,JVM崩溃问题得以解决:
1)Native.setProtected(true);
2)使用Native库释放内存 Native.free(Pointer.nativeValue(my_pointer_by_ref.getValue()));
3)使用以下CLI选项运行程序 - -XX:-UseLoopPredicate -DLD_PRELOAD = / usr / lib / jvm / java-8-oracle / jre / lib / amd64 / libjsig.so