System.loadLibrary不再作为Java程序包的一部分工作

时间:2018-01-16 10:21:49

标签: java c++ apache-spark java-native-interface swig

我想在使用Java运行的Apache Spark中使用现有的C ++代码。我使用SWIG生成JNI接口,只要我从这样一个简单的类(MyJavaClass.java)调用函数,它就像魅力一样:

public class MyJavaClass {
  public static void main(String[] args) {
    System.loadLibrary("mycppcode");
    mycppcode.dostuff();
  }
}

然后我尝试将该类集成到我的包装层中,我需要能够从Spark(MyJavaClass.java)调用它:

package com.mycompany.mypackage;

public class MyJavaClass {
  public void MyJavaMethod() {
    System.loadLibrary("mycppcode");
    mycppcode.dostuff();
  }
}

但是,只要我添加行,我就会在文件上运行 javac 时收到以下错误:

MyJavaClass.java:6: error: cannot find symbol
        mycppcode.dostuff();
        ^
  symbol:   variable mycppcode
  location: class MyJavaClass
1 error

我使用以下命令生成JNI,libmycppcode.so库和Java类:

swig -c++ -java -package com.mycompany.mypackage mycppcode.i
g++ -fpic -c mycppcode.cpp mycppcode_wrap.cxx -std=c++17 -I/usr/lib/jvm/java-8-openjdk-amd64/include -I/usr/lib/jvm/java-8-openjdk-amd64/include/linux
g++ -shared mycppcode.o mycppcode.o -o libmycppcode.so
javac MyJavaClass.java

这是C ++文件(mycppcode.cpp):

#include <iostream>
using namespace std;

int dostuff() {
    cout << "Hello World!" << endl;
}

这是接口文件(mycppcode.i):

%module mycppcode
%{
extern int dostuff();
%}
extern int dostuff();

这是生成的JNI文件(mycppcodeJNI.java):

package com.mycompany.mypackage

public class mycppcodeJNI {
  public final static native int dostuff();
}

这是生成的Java文件(mycppcode.java):

public class mycppcode {
  public static int dostuff() {
    return mycppcodeJNI.dostuff();
  }
}

2 个答案:

答案 0 :(得分:1)

我刚开始工作了。非常感谢@Michael提供 -package 参数的提示。

问题似乎是我试图从类文件的实际文件夹中运行 javac ,如下所示:

~/myproject/src/com/mycompany/mypackage$ javac MyJavaClass.java

javac 需要从根源目录执行,如下所示:

~/myproject/src$ javac com/mycompany/mypackage/MyJavaClass.java

我不确定为什么,但这可以防止错误。

答案 1 :(得分:0)

正如here所述,您无法在 com.mycompany.mypackage中使用默认包中的类。 MyJavaClass 即可。您应该重命名 mycppcode 类:

package com.mycompany.mypackage;

public class mycppcode {
  public static int dostuff() {
    return mycppcodeJNI.dostuff();
  }
}

实际上,它可能使用任何包名称,例如

package jni;

public class mycppcode {
  public static int dostuff() {
    return mycppcodeJNI.dostuff();
  }
}