JNA调用dll的问题
这个ASIMLib.DLL里面还包含两个对象,IM和MsgASIMLib.IMim=newIM();im.Init("","","");ASIMLib.Msgmsg=ne...
这个ASIMLib.DLL里面还包含两个对象,IM和Msg
ASIMLib.IM im = new IM();
im.Init("", "","");
ASIMLib.Msg msg = new Msg();
msg.Subject = "";
msg.Body = "";
im.SendMsgEx(msg, "");
使用JNA(Java Native Access)应该如何实现动态调用DLL中的对象和方法 展开
ASIMLib.IM im = new IM();
im.Init("", "","");
ASIMLib.Msg msg = new Msg();
msg.Subject = "";
msg.Body = "";
im.SendMsgEx(msg, "");
使用JNA(Java Native Access)应该如何实现动态调用DLL中的对象和方法 展开
2个回答
展开全部
例子:
JNA 把一个 dll/.so 文件看做是一个 Java 接口。
Dll 是 C 函数的集合、容器,这正和接口的概念吻合。
我们定义这样一个接口,
public interface TestDll1 extends Library {
/**
* 当前路径是在项目下,而不是 bin 输出目录下。
*/
TestDll1 INSTANCE = (TestDll1)Native.loadLibrary("TestDll1", TestDll1.class);
public void say(WString value);
}
如果 dll 是以 stdcall 方式输出函数,那么就继承 StdCallLibrary 。否则就继承默认的 Library 接口。
接口内部需要一个公共静态常量: instance 。
TestDll1 INSTANCE = (TestDll1)Native.loadLibrary("TestDll1", TestDll1.class);
通过这个常量,就可以获得这个接口的实例,从而使用接口的方法。也就是调用外部 dll 的函数!
注意:
1 , Native.loadLibrary() 函数有 2 个参数:
1 , dll 或者 .so 文件的名字,但不带后缀名。这符合 JNI 的规范,因为带了后缀名就不可以跨操作系统平台了。
搜索 dll 的路径是:
1 )项目的根路径
2 )操作系统的全局路径、
3 ) path 指定的路径。
2 ,第二个参数是本接口的 Class 类型。
JNA 通过这个 Class 类型,根据指定的 dll/.so 文件,动态创建接口的实例。
2 ,接口中你只需要定义你需要的函数或者公共变量,不需要的可以不定义。
public void say(WString value);
参数和返回值的类型,应该和 dll 中的 C 函数的类型一致。
这是 JNA ,甚至所有跨平台调用的难点。
JNA 把一个 dll/.so 文件看做是一个 Java 接口。
Dll 是 C 函数的集合、容器,这正和接口的概念吻合。
我们定义这样一个接口,
public interface TestDll1 extends Library {
/**
* 当前路径是在项目下,而不是 bin 输出目录下。
*/
TestDll1 INSTANCE = (TestDll1)Native.loadLibrary("TestDll1", TestDll1.class);
public void say(WString value);
}
如果 dll 是以 stdcall 方式输出函数,那么就继承 StdCallLibrary 。否则就继承默认的 Library 接口。
接口内部需要一个公共静态常量: instance 。
TestDll1 INSTANCE = (TestDll1)Native.loadLibrary("TestDll1", TestDll1.class);
通过这个常量,就可以获得这个接口的实例,从而使用接口的方法。也就是调用外部 dll 的函数!
注意:
1 , Native.loadLibrary() 函数有 2 个参数:
1 , dll 或者 .so 文件的名字,但不带后缀名。这符合 JNI 的规范,因为带了后缀名就不可以跨操作系统平台了。
搜索 dll 的路径是:
1 )项目的根路径
2 )操作系统的全局路径、
3 ) path 指定的路径。
2 ,第二个参数是本接口的 Class 类型。
JNA 通过这个 Class 类型,根据指定的 dll/.so 文件,动态创建接口的实例。
2 ,接口中你只需要定义你需要的函数或者公共变量,不需要的可以不定义。
public void say(WString value);
参数和返回值的类型,应该和 dll 中的 C 函数的类型一致。
这是 JNA ,甚至所有跨平台调用的难点。
追问
你没看清楚问题 我问的是如果dll中是C++对象中包含对象的话该如何映射到java代码中来
展开全部
首先你得写一个和C++中对应的java类。由于我不知道IM和Msg这两个对象的具体情况,只能给你举个例子:这是模拟的C的File类。
public interface MyCDll extends Library {
MyCDll INSTANCE = (MyCDll)Native.loadLibrary("MyCDll", MyCDll.class);
public static class CFile extends Structure {
public ByteByReference _ptr = new ByteByReference();
public int _cnt;
public ByteByReference _base = new ByteByReference();
public int _flag;
public int _file;
public int _charbuf;
public int _bufsiz;
public ByteByReference _tmpfname = new ByteByReference();
public static class ByReference extends CFile
implements Structure.ByReference{}
public static class ByValue extends CFile
implements Structure.ByValue{}
}
}
如果你要用到对象本身,这样用:
CFILE.ByValue f = new CFILE.ByValue();你上面的IM对象可以这么声明
如果你要作为把对象作为参数,这样用:
CFILE.ByReference f = new CFILE.ByReference ();你上面的Msg对象可以这么声明
public interface MyCDll extends Library {
MyCDll INSTANCE = (MyCDll)Native.loadLibrary("MyCDll", MyCDll.class);
public static class CFile extends Structure {
public ByteByReference _ptr = new ByteByReference();
public int _cnt;
public ByteByReference _base = new ByteByReference();
public int _flag;
public int _file;
public int _charbuf;
public int _bufsiz;
public ByteByReference _tmpfname = new ByteByReference();
public static class ByReference extends CFile
implements Structure.ByReference{}
public static class ByValue extends CFile
implements Structure.ByValue{}
}
}
如果你要用到对象本身,这样用:
CFILE.ByValue f = new CFILE.ByValue();你上面的IM对象可以这么声明
如果你要作为把对象作为参数,这样用:
CFILE.ByReference f = new CFILE.ByReference ();你上面的Msg对象可以这么声明
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
推荐律师服务:
若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询