您的位置:首页 > 编程语言 > C语言/C++

【转载】java调用C++写的DLL

2014-03-31 16:15 423 查看
用java调用C++写的DLL一直以来都是一个比较麻烦但又很常见的问题。

我们知道,使用 JNI 调用 .dll/.so 共享类库是非常非常麻烦和痛苦的。

如果有一个现有的 .dll/.so 文件,如果使用 JNI 技术调用,我们首先需要另外使用 C 语言写一个 .dll/.so 共享库,使用 SUN 规定的数据结构替代 C 语言的数据结构,调用已有的 dll/so 中公布的函数。然后再在 Java 中载入这个适配器 dll/so ,再编写 Java native 函数作为 dll 中函数的代理。经过 2 个繁琐的步骤才能在 Java 中调用本地代码。因此,很少有 Java 程序员愿意编写调用 dll/.so 库中的原生函数的 java 程序。这也使 Java 语言在客户端上乏善可陈。可以说 JNI 是 Java 的一大弱点。

现在好了出现了一个JNA的武器,JNA框架是一个开源的 Java 框架,是 SUN 公司主导开发的,建立在经典的 JNI 的基础之上的一个框架。通过JNA能够很方便的调用C++写的DLL。

先来一个Hello Word

C++代码

#ifndef DEMOC_H
#define DEMOC_H
#ifndef DLLDIR
#define DLLDIR  extern "C"__declspec(dllexport)
#endif
typedef struct{
char *m_sername;
char *m_username;
char *m_password;
WORD m_tranType;
HWND m_hChMsgWnd;
UINT m_nChmsgid;
int  m_sockType;
int  m_errormsg;
void (WINAPI *m_messagecallback)(LONG hHandle,char *wParam,int lParam);
void *context;
}SERVER_UPDATEINFO;
DLLDIR LONG __stdcall VSNET_ServerStart(char *m_url,SERVER_UPDATEINFO *m_pSerinfo,WORD wserport = 3000);
#endif


这是一个标准的C++头文件,在这个头文件中定义了一个函数VSNET_ServerStart,下面来分析一下这个头文件。首先看到在这个头文件中定义了一个宏DLLDIR,这个宏意义为导出编码为标准C(extern "C"),并且为declspec的。

然后看到在头文件中还定义了一个结构体SERVER_UPDATEINFO,在SERVER_UPDATEINFO中具有以下几种数据类型:char类型指针,word类型,hwnd类型,uint,int,回调函数,void指针。

这就是我们知道DLL中的所有信息。

对了还有一个比较关键的地方,在C、C++中如果直接编译DLL,在DLL中导出的函数名是会在编译时被编译器修改掉,所以我们必须制定导出的函数名,这就需要在工程中定义一个def文件。

def文件

LIBRARY    "demoC"
EXPORTS
VSNET_ServerStart


好了到此为止这就是我们所知道的所有DLL信息,一般这也是DLL组件会暴漏出来的所有信息。下面就来说明用JNA如何调用。

首先我们先把DLL中的函数导出来,编程JAVA 的代码

public interface CLibrary extends StdCallLibrary {
VSNET_ServerStart
}


声明一个CLibrary的java接口,继承自JNA类库中的StdCallLibrary。OK已经导出来了,对了就这么简单。

那么有些同学就问了,这个函数在C++的头文件中有一个LONG的返回值,而且还有参数啊?别急下面就来讲解。

首先我们看到VSNET_ServerStart有一个LONG的返回值,LONG是C++中的数据类型,在java中是没有的,但是JNA已经为我们提供了这种类型叫做NativeLong。OK那么修改一下我们的JAVA代码

public interface CLibrary extends StdCallLibrary {
NativeLong VSNET_ServerStart
}


下面是JNA类型与C++类型的对比:



接下来我们来看函数的参数如何处理。

VSNET_ServerStart函数的参数为(char *m_url,SERVER_UPDATEINFO *m_pSerinfo,WORD wserport = 3000)。通过类型的对比我们知道char *=String,WORD=short。那个SERVER_UPDATEINFO这种结构体如何处理呢。

在JNA中类库提供了一种Structure的类型专门处理结构体。下面我们用JAVA来描述一下这个结构体。

public static class SERVER_UPDATEINFO extends Structure {
public String m_sername;
public String m_username;
public String m_password;
public short m_tranType;
public int m_hChMsgWnd;
public int m_nChmsgid;
public int m_sockType;
public int m_errormsg;
public ICallback fun;
}


对函数指针的处理

对函数指针的处理

public interface ICallback extends StdCallCallback{
public abstract void fun(NativeLong hHandle,String wParam,int lParam);
}

OK我们成功的描述的结构体,下面再来修改我的java接口。

public interface CLibrary extends StdCallLibrary {
NativeLong VSNET_ServerStart(String m_url,SERVER_UPDATEINFO s,short sho)
}

到此就全部完成了DLL到JAVA的导出。下面来看如何使用。

CLibrary INSTANCE = (CLibrary)Native.loadLibrary("demoC", CLibrary.class);
ICallback C = new CallbackImp();
CLibrary.SERVER_UPDATEINFO su = new CLibrary.SERVER_UPDATEINFO();
su.m_sername = "sername";
su.m_username = "username";
...
su.fun = C;
short sho = 3000;
NativeLong nl = CLibrary.INSTANCE.VSNET_ServerStart("url",su,sho);

OK,搞定。

注意: 1、函数、结构体定义的名称必须和DLL中的一样。 2、函数结构体的参数顺序必须一样
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: