有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

dll JNA线程“main”java中出现异常。lang.错误:无效的内存访问(未知源)

我正在使用JNA 4.0.0从Java访问一些DLL函数,该DLL本机函数声明如下:

int ApplicationInit(HANDLE hEMV, TLV *tlv_Appl, TLV *tlv_AIP);

输入参数的类型如下所述

/* Opaque structure */
typedef void *HANDLE;

typedef struct
{
    unsigned char *_lenptr;     /* pointer to 'len' field (Private member) */
    unsigned int _len;          /* 'outer' length, specified by user (Private member) */
    unsigned short _offset;
    unsigned short len;         /* number of bytes (Public member) */

    unsigned long tag;          /* Tag tag  (Public member) */
    unsigned char *val;         /* byte string  (Public member) */
    unsigned char _tagptr[256]; /* Container for TLV data (Private member) */
} TLV;

因此,我在库接口中声明了它,如下所示:

public static class HANDLE extends PointerType {
        public HANDLE(Pointer address) {
            super(address);
        }
        public EMV_HANDLE() {
            super();
        }
    }

public class TLV extends Structure {
    /**
     * pointer to 'len' field (Private member)<br>
     * C type : unsigned char*
     */
    public Pointer _lenptr;
    /** 'outer' length, specified by user (Private member) */
    public int _len;
    public short _offset;
    /** number of bytes (Public member) */
    public short len;
    /** Tag tag  (Public member) */
    public NativeLong tag;
    /**
     * byte string  (Public member)<br>
     * C type : unsigned char*
     */
    public Pointer val;
    /**
     * Container for TLV data (Private member)<br>
     * C type : unsigned char[256]
     */
    public byte[] _tagptr = new byte[256];
    public TLV() {
        super();
    }
    protected List<? > getFieldOrder() {
        return Arrays.asList("_lenptr", "_len", "_offset", "len", "tag", "val", "_tagptr");
    }
    /**
     * @param _lenptr pointer to 'len' field (Private member)<br>
     * C type : unsigned char*<br>
     * @param _len 'outer' length, specified by user (Private member)<br>
     * @param len number of bytes (Public member)<br>
     * @param tag Tag tag  (Public member)<br>
     * @param val byte string  (Public member)<br>
     * C type : unsigned char*<br>
     * @param _tagptr Container for TLV data (Private member)<br>
     * C type : unsigned char[256]
     */
    public TLV(Pointer _lenptr, int _len, short _offset, short len, NativeLong tag, Pointer val, byte _tagptr[]) {
        super();
        this._lenptr = _lenptr;
        this._len = _len;
        this._offset = _offset;
        this.len = len;
        this.tag = tag;
        this.val = val;
        if ((_tagptr.length != this._tagptr.length)) 
            throw new IllegalArgumentException("Wrong array size !");
        this._tagptr = _tagptr;
    }
    public static class ByReference extends TLV implements Structure.ByReference {

    };
    public static class ByValue extends TLV implements Structure.ByValue {

    };
}

int EMV_ApplicationInit(AppdefLibrary_EMVCT.EMV_HANDLE hEMV, TLV.ByReference tlv_Appl, TLV.ByReference tlv_AIP);

然后我用下面的方式来称呼它:

EMV_HANDLE hEMV= new EMV_HANDLE();
TLV.ByReference tlv_Appl=new TLV.ByReference();
TLV.ByReference tlv_AIP=new TLV.ByReference();

System.out.println(AppdefLibrary.INSTANCE.ApplicationInit(hEMV, tlv_Appl, tlv_AIP));

但我得到了以下例外:

Exception in thread "main" java.lang.Error: Invalid memory access
    at com.sun.jna.Native.invokeInt(Native Method)
    at com.sun.jna.Function.invoke(Function.java:383)
    at com.sun.jna.Function.invoke(Function.java:315)
    at com.sun.jna.Library$Handler.invoke(Library.java:212)
    at com.sun.proxy.$Proxy1.ApplicationInit(Unknown Source)
    at test.Test.main(Test.java:192)

请帮助并感谢您的关注


共 (1) 个答案

  1. # 1 楼答案

    当您试图访问未正确分配的本机端内存时,会出现无效内存访问错误

    内存分配的具体方式有几种,你必须找出它们来解决你的问题。。。你必须剥下几层洋葱才能解决实际问题

    首先要检查的是JNA类型映射。结构尺寸在这里是出了名的错误。但是,您的结构看起来是正确的

    下一个可能的原因是您没有为结构本身分配本机端内存;这是您选择使用ByReference处理结构的一个副作用。如果你走这条路,你会有更多的开销。但这都是不必要的,;JNA在幕后完成所有繁重的工作,当您将结构作为参数传递给JNA库时,它实际上只发送指向本机端的指针。。。但是处理内存分配等

    您应该直接在代码中引用TLV结构

    图书馆:

    int EMV_ApplicationInit(AppdefLibrary_EMVCT.EMV_HANDLE hEMV, TLV tlv_Appl, TLV tlv_AIP);
    

    在您的访问代码中:

    TLV tlv_Appl=new TLV();
    TLV tlv_AIP=new TLV();
    
    System.out.println(AppdefLibrary.INSTANCE.ApplicationInit(hEMV, tlv_Appl, tlv_AIP));
    

    如果解决这个问题不能解决问题,另一种可能性是,当您为Java结构分配了内存时,C方法在内部引用其他内存,API希望您以某种方式初始化变量,以便它们指向(分配的)其他内存。未初始化的指针很可能是罪魁祸首,特别是当它是用户的“输入”,而不是函数将填充的回调时。您需要仔细查看API文档,看看是否存在这种情况,以及是否需要初始化任何指针

    例如,空HANDLE可能是该方法不可接受的;它可能需要您用其他方法初始化HANDLE(然后再释放它)

    或者,您可能需要对TLV结构执行一些操作来初始化其内部指针成员。例如,val字段指向一个“字节字符串”。API是否期望您已经分配了该内存并提供了分配的内存长度(例如Pointer val = new Memory(256); tlv_Appl.val = val; tlv_Appl.len = 256;)?_lenptr到底指向什么(注释表示一个int字段,但它是一个char指针,这似乎很奇怪)

    另外,顺便说一句,4.0.0是JNA的一个更老的版本;4.2.2是当前版本,除非出于其他原因需要旧版本,否则应使用该版本

    另一方面,JNA已经包含了WinNT.HANDLE的映射,这可能比您自己的映射更好