JNA 4.2.1 调用不带参数的 dll 方法

JNA 4.2.1 call dll method without params

我用的是jna 4.2.1 我在 dll 中有一个方法 returns 一个指向对象的指针 (C++)

basic_hash* getAlgorithmInstance( int algorithm )

basic_hash 有以下方法(C++):

void reset ();
void partial (const byte* data, uint64 size);
void finalize (vector_byte& hash);
void hash (const byte* data, uint64 size, vector_byte& hash).

我有接口 (java)

public interface HAL extends Library {
  HAL INSTANCE = (HAL) Native.loadLibrary(
    (Platform.isWindows() ? "HAL" : "libHAL"), HAL.class);
  BasicHash getAlgorithmInstance(int i);
}
public static class BasicHash extends Structure {
  public BasicHash() {}
  public BasicHash(Pointer p) {
    super(p);
    read();
  }
  @Override
  protected List getFieldOrder() {
    return Arrays.asList(new String[] { "reset", "hash", "partial", "finalize" });
  }
  public interface Reset extends Callback { public void invoke();}
  public Reset reset;
  public interface Hash extends Callback {public void invoke(byte[] data, long size, byte[] hash);}
  public Hash hash;
  public interface Partial extends Callback {public void invoke(Pointer data, long size);}
  public Partial partial;
  public interface Finalize extends Callback {public void invoke(byte[] hash);}
  public Finalize finalize;
}

当我在 main() 中使用不带参数的方法时

HAL lib = HAL.INSTANCE;
BasicHash b = lib.getAlgorithmInstance(0);
b.reset.invoke();

我得到一个错误:

Exception in thread "main" java.lang.Error: Invalid memory access
  at com.sun.jna.Native.invokeVoid(Native Method)
  at com.sun.jna.Function.invoke(Function.java:374)
  at com.sun.jna.Function.invoke(Function.java:323)
  at com.sun.jna.Function.invoke(Function.java:275)
  at com.sun.jna.CallbackReference$NativeFunctionHandler.invoke(CallbackReference.java:646)
  at com.sun.proxy.$Proxy1.invoke(Unknown Source)
  at net.erver.ItServer.main(ItServer.java:79)

如果方法重置库中的变量,为什么我会收到此错误?该方法本身可以毫无问题地实现(根据开发人员 dll)

编辑: vector_byte 有定义:

typedef unsigned char byte;
typedef std::vector< byte > vector_byte

和basic_hash有定义:

namespace HAL { namespace algorithms {
  HAL_HASH_API enum class State : byte {
    Normal,
    Finished,
  };
  class HAL_HASH_API basic_hash 
  {
    public:
    virtual ~basic_hash() {}
    virtual void reset() = 0;
    virtual void partial( const byte*, uint64 ) = 0;
    virtual void finalize( vector_byte& ) = 0;
    virtual void hash( const byte*, uint64, vector_byte& ) = 0;

    bool isFinished() {
      return ( _state == State::Finished ? true : false );
    }
    protected:
      State _state;
  };
}
}

您需要使用普通 struct 在 JNA 和您的库之间传递数据。 C++ class(包括 vector 模板)的内存布局与简单的 C struct.

有很大不同