java - 如何在不出现 UnsatifiedLinkError 的情况下加载具有依赖项的 JNI .dylib 文件?

标签 java c swift java-native-interface

目标:将 Java 链接到 Swift

问题:我得到一个 UnsatisfiedLinkError调用 System#loadLibrary(String) 时尝试加载与 Swift .dylib 文件链接的 JNI .dylib 文件时.

预期行为:Java .dylib 的依赖项将被自动加载或调用 System.loadLibrary("SwiftCode")会加载依赖项(我能想到的唯一解决方案)。

注意:我合并了this github tutorialthis Medium article创建我的 JNI .dylib 文件和 this tutorial创建我的 Swift .dylib 文件。

完整的堆栈跟踪:

Exception in thread "main" java.lang.UnsatisfiedLinkError: /Users/hillmacbookpro/IdeaProjects/JavaToSwift/src/native/libSwiftHelloWorld.dylib: dlopen(/Users/hillmacbookpro/IdeaProjects/JavaToSwift/src/native/libSwiftHelloWorld.dylib, 1): Library not loaded: libSwiftCode.dylib
  Referenced from: /Users/hillmacbookpro/IdeaProjects/JavaToSwift/src/native/libSwiftHelloWorld.dylib
  Reason: image not found
    at java.base/java.lang.ClassLoader$NativeLibrary.load0(Native Method)
    at java.base/java.lang.ClassLoader$NativeLibrary.load(ClassLoader.java:2408)
    at java.base/java.lang.ClassLoader$NativeLibrary.loadLibrary(ClassLoader.java:2465)
    at java.base/java.lang.ClassLoader.loadLibrary0(ClassLoader.java:2662)
    at java.base/java.lang.ClassLoader.loadLibrary(ClassLoader.java:2627)
    at java.base/java.lang.Runtime.loadLibrary0(Runtime.java:829)
    at java.base/java.lang.System.loadLibrary(System.java:1833)
    at helloworld.SwiftHelloWorld.<clinit>(SwiftHelloWorld.java:7)

文件结构:

src
  - helloworld
      - SwiftHelloWorld.java
  - native
      - libSwiftHelloWorld.dylib
      - libSwiftCode.dylib

MCVE:

SwiftHelloWorld.java:

package helloworld;

public class SwiftHelloWorld {

    static {
        System.loadLibrary("SwiftCode"); // loading SwiftHelloWorld's dependency first
        System.loadLibrary("SwiftHelloWorld"); // exception thrown here, can't find dependency?
    }

    public static native void printHelloWorldImpl();

    public static void main(final String[] args) {
        printHelloWorldImpl();
    }

}

libSwiftHelloWorld.dylib:使用这两个终端命令创建:

export JAVA_HOME="$(/usr/libexec/java_home -v 11)"; gcc -I"$JAVA_HOME/include" -I"$JAVA_HOME/include/darwin/" -o libSwiftHelloWorld.dylib -dynamiclib helloworld_SwiftHelloWorld.c libSwiftCode.dylib

和这些文件:

  • helloworld_SwiftHelloWorld.c:

    #include <jni.h>
    #include <stdio.h>
    #include "helloworld_SwiftHelloWorld.h"
    #include "helloworld_SwiftHelloWorld_swift.h"
    
    JNIEXPORT void JNICALL Java_helloworld_SwiftHelloWorld_printHelloWorldImpl
      (JNIEnv *env, jclass clazz) {
    
        int result = swiftHelloWorld(42);
        printf("%s%i%s", "Hello World from JNI! ", result, "\n");
    
    }
    
  • helloworld_SwiftHelloWorld.h:自动生成的 JNI 头文件 javac -h native/helloworld/SwiftHelloWorld.java

  • helloworld_SwiftHelloWorld_swift.h:

    int swiftHelloWorld(int);
    

libSwiftCode.dylib:使用 swiftc SwiftCode.swift -emit-library -o libSwiftCode.dylibSwiftCode.swift 文件创建:

import Foundation

// force the function to have a name callable by the c code
@_silgen_name("swiftHelloWorld")
public func swiftHelloWorld(number: Int) -> Int {
    print("Hello world from Swift: \(number)")
    return 69
}

相关问题:

  • Java JNI linking multiple libraries ,他/她得到一个 UnsatifiedLinkError 但带有描述 undefined symbol: ... (unanswered)

  • 任何带有 JNI 标签并包含 UnsatifiedLinkError

其他注意事项:我将 Java 虚拟机的选项设置为 -Djava.library.path=src/native,这两个 的父目录.dylib 文件。我正在使用 macOS。

编辑: 我尝试在命令之前使用 xcrun 编译 SwiftCode.swift,如 this article 中所示(在“编译 Swift 代码”下)但我仍然遇到相同的错误。

最佳答案

macOS ld将库依赖项的路径构建到二进制文件中。装载机装载libSwiftHelloWorld.dylib只会找到 libSwiftCode.dylib如果后者在当前目录中。在 Java 中加载依赖项是行不通的,因为对于加载程序来说,它是一个不同的库。

您可以更改 libSwiftCode.dylib 的内置路径与 -install_name参数(即 swiftc ... -Xlinker -install_name -Xlinker <your path> )。如果重建 libSwiftHelloWorld.dylib之后它将引用您提供的路径。

关于java - 如何在不出现 UnsatifiedLinkError 的情况下加载具有依赖项的 JNI .dylib 文件?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53137790/

相关文章:

c - 将已删除的文件重新附加到目录

c - fseek 在 Linux 中不起作用

ios - 重新打开游戏时执行“追赶”

java - 在android中将数据从一个文件复制到另一个文件非常慢?

java - 二进制转十进制,错误信息

java - 如何使 ImageView 在选项卡 JavaFx 中居中?

c++ - 为什么显式位移运算会产生更大的 .s 文件?

ios - 创建一个包含 n 个空格或其他重复字符的字符串

ios - 调用中的额外参数 "animated"- 错误

java - while 循环内不会打印行