0

我目前正在尝试使用来自 Java 的 NTX 索引访问 DBF。我有一个多年前的 Artemis 引擎(现在是 ApolloDB)的副本,它可以在 VB6 中执行此操作。它主要使用 3 个 DLL,主要是 SDE32.DLL。

我已经使用 NativeCall 成功访问了这些 DLL 中的许多函数

public static VoidCall sx_Zap = new VoidCall("SDE32", "sx_Zap"); public static IntCall sx_Use = new IntCall("SDE32", "sx_Use");

intFile = sx_Use.executeCall(fileName);
        if (intFile == 0){
            if (JOptionPane.showOptionDialog(null, "Could not open:" + fileName + "\nRetry?", "Failed to open DBF", JOptionPane.YES_NO_OPTION, JOptionPane.QUESTION_MESSAGE, null, null, null) != JOptionPane.YES_OPTION){
                return;
            }else{
                sx_Zap.executeCall();
            }
        }

该代码将非常高兴地打开并删除数据库,因此我知道我在正确的轨道上。请注意,当我想返回一个整数时,我使用了 IntCall,而当我不想返回任何内容时,我使用了 VoidCall。

我发现的问题是一些函数,如一些数据访问函数返回字符串,例如 VB6 中的 sx_GetString 函数的函数声明,该函数从当前记录中获取一个字符串字段。 Declare Function sx_GetString Lib "sde32.dll" (ByVal cpFieldName As String) As String

如何将这些信息导入 Java?似乎只有 int(和 boolean)和 void 返回类型,我怎样才能得到 String、double 和 long 类型?

使用 JNA,我似乎可以访问双精度和长整数,但是当我尝试使用字符串返回类型时,我得到了执行保护违规并且 Java 崩溃了。

JNA 示例:

public interface SDE32 extends Library {
   public String sx_GetString(String cpFieldName);
   public Double sx_GetDouble(String cpFieldName);
   public Long sx_GetLong(String cpFieldName);

}

JNADBF.SDE32 sde = (JNADBF.SDE32) Native.loadLibrary("SDE32", JNADBF.SDE32.class);
    System.out.println(sde.sx_GetString("TILLNAME"));
    System.out.println(sde.sx_GetDouble("SELLPRICE"));

JNA 错误:

Execution protection violation

#

# A fatal error has been detected by the Java Runtime Environment:

#

# EXCEPTION_ACCESS_VIOLATION (0xc0000005) at pc=0x00000000, pid=11104, tid=11060

那么我怎样才能得到这些返回类型呢?尤其是字符串。

非常感激任何的帮助。

4

1 回答 1

1

我建议首先返回一个指针(而不是字符串),然后使用各种指针方法检查它指向的内存。一旦了解了函数返回的内容,您就可以弄清楚如何使 JNA 正确恢复数据(自动或显式)。

编辑

public interface SDE32 extends StdCallLibrary {
   public Pointer sx_GetString(String cpFieldName); // don't use String just yet
   public double sx_GetDouble(String cpFieldName);
   // Don't use Java "long" unless you want a 64-bit integer
   // On windows, native "int" and "long" are both 32 bits
   public NativeLong sx_GetLong(String cpFieldName);
}

编辑 参考更新版本的pascal 头文件,您应该使用 Java 类型shortforsmallIntWordBool,和intfor Long。但是,切换到这些类型并不能提供完整的修复(它将减少堆栈损坏 - 您将使用错误的标志“打开”数据库)。

于 2012-09-04T14:23:49.007 回答