2

この質問で既に説明したように、必要な JNI バインディングを作成するためにJavaCPPを使用して、Java から Haskell コードを呼び出そうとしています。

これは私がそれを使用している方法です:

<rootdir>
  /javacpp.jar
  /build (destination of libraris)
  /src   (contains Haskell code)
  /com/example/HSCode.java (Java class to load and use native lib)

の内容HScode.java:

package com.example;

import org.bytedeco.javacpp.*;
import org.bytedeco.javacpp.annotation.*;
import java.util.Scanner;
import java.io.File;
import java.io.FileNotFoundException;

@Platform(include={"<HsFFI.h>","HScode_stub.h"})
public class HScode {
    static { Loader.load(); }
    public static native void hs_init(int[] argc, @Cast("char***") @ByPtrPtr PointerPointer argv);
    public static native String code_hs(String text);

    public static void main(String[] args) throws FileNotFoundException {
        String s = new Scanner(new File("test.txt")).useDelimiter("\\Z").next();
        hs_init(null, null);
        String s1 = code_hs(s);
        System.out.println(s1);
    }
}

コンパイル:

cd <rootdir>
ghc --make -isrc -dynamic -shared -fPIC src/HScode.hs \
     -o build/libHScode.so -lHSrts-ghc7.8.4 -optl-Wl,-rpath,.
javac -cp javacpp.jar com/example/HScode.java
java -jar javacpp.jar -d build \
     -Dplatform.compiler=ghc -Dplatform.includepath="src:com/example" \
     -Dplatform.compiler.output="-optl-Wl,-rpath,. -optc-O3 -Wall build/libHScode.so -dynamic -fPIC -shared -lstdc++ -lHSrts-ghc7.8.4 -o " com.example.HScode

このアプローチに従って、 alibHScode.solibjniHScode.sousing を作成できますjavacpp

$ java -cp javacpp.jar:. com.example.HScode

ジャー

さて、次のステップは、すべてを jar にパッケージ化し、この jarcom.example.HScodeをより大きな Java プロジェクトから使用できるようにすることです。

JavaCPP のページには次のように記載されています。

[...] さらに、実行時に、Loader.load() メソッドは、ビルド プロセスによって適切なディレクトリに配置された Java リソースからネイティブ ライブラリを自動的にロードします。JAR ファイルにアーカイブすることもできますが、何も変わりません。ユーザーは、システムにファイルをロードさせる方法を理解する必要はありません。

だから私はこれがうまくいくはずだと思った。

ただし、 上記のフォルダーHScode.jarのコンテンツからjar を作成すると、jar にとの両方が含まれ、次のように実行されます。buildlibjniHScode.solibHScode.so

$ java -cp javacpp.jar:HScode.jar:. com.example.HScode

その後、ネイティブコードが見つかりません(匿名化のために編集された例外):

Exception in thread "main" java.lang.UnsatisfiedLinkError: no jniHScode in java.library.path
    at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1865)
    at java.lang.Runtime.loadLibrary0(Runtime.java:870)
    at java.lang.System.loadLibrary(System.java:1122)
    at org.bytedeco.javacpp.Loader.loadLibrary(Loader.java:597)
    at org.bytedeco.javacpp.Loader.load(Loader.java:438)
    at org.bytedeco.javacpp.Loader.load(Loader.java:381)
    at com.example.HScode.<clinit>(HScode.java:13)
Caused by: java.lang.UnsatisfiedLinkError: /compilation-path/linux-x86_64/libjniHScode.so: HScode.so: cannot open shared object file: No such file or directory
    at java.lang.ClassLoader$NativeLibrary.load(Native Method)
    at java.lang.ClassLoader.loadLibrary0(ClassLoader.java:1937)
    at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1822)
    at java.lang.Runtime.load0(Runtime.java:809)
    at java.lang.System.load(System.java:1086)
    at org.bytedeco.javacpp.Loader.loadLibrary(Loader.java:580)

私は何が欠けていますか?JavaCPP が jar にアーカイブされたときにネイティブ コードを実際に見つけることができるかどうかを知っている人はいますか?

4

1 に答える 1

3

javacpp -jar javacpp.jar com.example.HScode出力をcom/example/linux-x86_64/自動的に呼び出してネイティブ ライブラリをビルドし、Loaderそこからロードします。そのため、他の方法でネイティブ ライブラリをビルドする場合でもcom/example/linux-x86_64/、JAR ファイル内にあるか、通常のファイルとして外部にあるかに関係なく、それらを に移動する必要がありLoaderます。

于 2016-01-25T11:14:50.637 に答える