我正在尝试使用 JNA 同时操作本机、非线程安全的 Fortran 库。由于该库不是线程安全的,我尝试实例化同一库的不同副本,但显然它们似乎共享内存地址。如果我修改一个库中的一个变量,则另一个库中的变量也会被修改。这种行为使得它们不可能在单独的线程中同时运行。
下面的代码示例演示了我的意思:
code.f:
subroutine set(var)
implicit none
integer var,n
common/conc/n
n=var
end subroutine
subroutine get(var)
implicit none
integer var,n
common/conc/n
var=n
end subroutine
该文件的编译和复制如下:
gfortran -shared -O2 code.f -o mylib.so -fPIC
cp mylib.so mylib_copy.so
然后我使用 JNA 访问这两个:
import com.sun.jna.Library;
import com.sun.jna.Native;
import com.sun.jna.ptr.IntByReference;
public class JNA {
public interface MyLib extends Library {
public void set_(IntByReference var);
public void get_(IntByReference var);
}
public static void main(String[] args) {
System.setProperty("jna.library.path", ".");
MyLib lib = (MyLib) Native.loadLibrary("mylib.so", MyLib.class);
MyLib lib_copy = (MyLib) Native.loadLibrary("mylib_copy.so", MyLib.class);
# set a common variable in mylib
lib.set_(new IntByReference(9));
# access the variable in mylib_copy
IntByReference result = new IntByReference();
lib_copy.get_(result);
System.out.println(result.getValue());
}
上述代码的输出是9
,这意味着这两个库似乎共享它们的内存。
有没有办法让他们完全独立呢?我使用 Intel Fortran 编译器进行了相同的尝试,得到了相同的结果。
JNA 在通过以下方式打开库时使用 RTLD_LAZY|RTLD_GLOBALdlopen
,这可能就是共享符号的原因。您可以像这样覆盖这些标志:
int RTLD_LOCAL = ??; // look this up on your system
Map options = new HashMap();
options.put(Library.OPTION_OPEN_FLAGS, RTLD_LOCAL);
MyLib mylib = Native.loadLibrary("mylib", MyLib.class, options);
MyLib mylib2 = Native.loadLibrary("mylib2", MyLib.class, options);
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)