我想在使用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();
}
}
答案 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();
}
}