进行 JNI 调用时出现 Unsatisfied Link 错误

Posted

技术标签:

【中文标题】进行 JNI 调用时出现 Unsatisfied Link 错误【英文标题】:Getting Unsatisfied Link error while making a JNI call 【发布时间】:2015-11-23 18:58:11 【问题描述】:

调用 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 错误?

本地方法中是否必须有 JNICall 和 JNIExport ?

如何将它与我的普通 c 文件集成?

【问题讨论】:

My .h file method is (Note not created from the Java class file ) 所以你没有使用javah.exe 创建 C 头文件? @PaulMcKenzie 是的,我没有使用 javah 创建头文件(基本上是在试验如何在不使用 javah 创建头文件的情况下调用本机方法) 堆栈跟踪demo.JNIWrapper.packet_mining_start(Ljava/lang/String;)I 描述的方法与您提供的Java 代码中声明的方法不匹配。它们的参数数量和类型不同。要么您遗漏了关键细节,要么您的课程处于不一致的状态。 @JohnBollinger 在本机方法中是否必须有 JNICall 和 JNIExport ?因为我怀疑 JNI 方法不包含 JNIExport 和 JNICall 它可能是问题,我的 JNI 方法是 uint32_t pax_packet_mining_start(uint8_t*挖掘查询文件名); 另外,您通常会给System.loadLibrary() 一个不带“.so”扩展名或路径的库名称。代替路径,您可以设置 LD_LIBRARY_PATH 以包含可以找到库的目录(在启动程序之前)。然而,也许 Java 足够聪明,可以理解您给出的形式。 【参考方案1】:

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 将无法找到函数。

【讨论】:

在 Linux 系统上,符号无论如何都会被导出,除非您将默认可见性更改为“隐藏”

以上是关于进行 JNI 调用时出现 Unsatisfied Link 错误的主要内容,如果未能解决你的问题,请参考以下文章

在使用jni调用so库时出现的找不到文件是怎么回事?

如何编写代码以使用 microsoft visual c++ 调用 JNI [关闭]

解决eclipse启动时出现“failed to load the jni shared library”

启动eclipse时出现“Failed to load the JNI shared library jvm.dll”错误及解决

解决安装eclipse时出现"Failed to load JNI shared library"

JNI_CreateJavaVM 上的段错误