用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++类型的对比:

c调用JAVA写的dll java c++ dll_c/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、函数结构体的参数顺序必须一样