我在调用 JNI 方法时收到 Unsatisfied link 错误。下面是我的日志
日志
new exception [Handler processing failed; nested exception is java.lang.UnsatisfiedLinkError: demo.JNIWrapper.packet_mining_start(Ljava/lang/String;)I] with root cause
java.lang.UnsatisfiedLinkError: demo.JNIWrapper.packet_mining_start(Ljava/lang/String;)I
at demo.JNIWrapper.packet_mining_start(Native Method)
at demo.RESTController.getPcap(RESTController.java:97)
下面是我的 JNI 代码
public class JNIWrapper {
static{
System.load("/home/XXX/XXX/Library/newJNI/libJNIDemo.so");
}
//Get Path in output param and status
public native int packet_mining_start(String fileName);
}
现在,从文件 Main.java 中调用此方法,并在为此类创建对象 (JNIWrapper jni=new JNIWrapper()) 期间调用 System.load 我没有收到任何错误。
现在,当我在 JNI 中调用 packet_mining_start 方法时,出现错误。
我的.h文件方法是(注意不是从Java类文件创建的)
uint32_t pax_packet_mining_start(uint8_t* mining_query_file_name);
此方法不包含 JNIExport 和 JNICall 变量或类型。因此,如果一个不包含 JNIExport 的方法并调用该方法会给出 UnsatisfiedLink 错误吗?
native 方法中是否必须包含 JNICall 和 JNIExport?
我如何将它与我的普通 c 文件集成?
最佳答案
JNIEXPORT 和 JNICALL 是根据您所在的平台定义的。如果您使用的是 Windows 系统,JNIEXPORT 在 DLL 中将是强制的,因为 DLL 包含有关导出哪些函数的信息,并且该函数在 DLL 外部不可见(因此您无法加载它)。在 Linux 系统上,符号无论如何都会导出,因此您不需要在它们上显式声明这一点。
JNICALL 是一种调用约定,在 Windows 上是 stdcall,在其他地方是 cdecl。如果您的调用约定不正确,您可能会损坏堆栈,从而导致段错误,但当时这似乎不是问题。
还有另一个问题,您似乎将 uint8_t* 作为参数,而您将 java String 对象传递给该方法。在java中,String是一种对象类型,它不会自动转换为C风格的const char*。
您必须在这两个函数之间有另一个函数,该函数接受 Java 参数、具有正确的调用约定并导出到库外部,然后将 Java String 对象转换为 uint8_t* 并调用原始函数。这样您就可以解决此错误(以及稍后可能会遇到的更多错误)。
既然您也将这个问题标记为 C++,请不要忘记将 extern“C” 放在函数前面,这样编译器就不会破坏名称。一旦函数的名称被破坏,Java 将无法找到它们。
关于java - 进行 JNI 调用时出现 Unsatisfied Link 错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33878523/