Java調用動態庫方法說明

Java不能直接調用由c或者c++寫得dll(TF_ID.dll),所以只能採用jni得方法,一步一步生成符合規範得dll文件(假設叫FANGJIAN.dll),在FANGJIAN.dll這個文件裏來調用TF_ID.dll。注意一點:兩個dll文件不能重名,爲什麼呢?因爲java後來執行時候,必須把兩個dll文件均考到java得class文件同一目錄下,或者把TF_ID.dll考到system32下也可以,如果重名得話,首先不能考到一個目錄下,再則即使將前一個dll考到system32下,那麼後一個dll也會出現調用自身dll得問題,大家可想而知了!!!
下面給出了兩個例子:

第一個是用java得jni方法生成了一個dll文件,這個dll文件,直接在其方法函數體內寫具體實現得方法,然後將dll文件考到java執行得同一目錄下,就可以執行成功了!

第二個也是用java得jni方法生成一個dll文件(FANGJIAN.dll),這個dll文件中再來調用IC卡讀寫器提供得dll文件(TF_ID.dll),將兩個dll文件考到java的class文件同一目錄下面,運行java文件就可以了!!!

(用jni生成一個C或者C++的dll,然後在那個dll裏面調用你說的這個dll,用java直接調用是不行的,因爲java調用的dll是必須準找一定的規則的,都是用javah生成本地方法的頭文件,然後寫c或着c++,然後編譯成dll)
例一:JAVA通過JNI調用本地C語言方法

JAVA以其跨平臺的特性深受人們喜愛,而又正由於它的跨平臺的目的,使得它和本地機器的各種內部聯繫變得很少,約束了它的功能。解決JAVA對本地操作的一種方法就是JNI。

JAVA通過JNI調用本地方法,而本地方法是以庫文件的形式存放的(在WINDOWS平臺上是DLL文件形式,在UNIX機器上是SO文件形式)。通過調用本地的庫文件的內部方法,使JAVA可以實現和本地機器的緊密聯繫,調用系統級的各接口方法。
簡單介紹及應用如下:

一、JAVA中所需要做的工作

在JAVA程序中,首先需要在類中聲明所調用的庫名稱,如下:
static {
System.loadLibrary(“goodluck”);
}
在這裏,庫的擴展名字可以不用寫出來,究竟是DLL還是SO,由系統自己判斷。

還需要對將要調用的方法做本地聲明,關鍵字爲native。並且只需要聲明,而不需要具體實現。如下:
public native static void set(int i);
public native static int get();
然後編譯該JAVA程序文件,生成CLASS,再用JAVAH命令,JNI就會生成C/C++的頭文件。

例如程序testdll.java,內容爲:
public class testdll{
    static{
        System.loadLibrary("goodluck");
    }
    public native static int get();
    public native static void set(int i);
    public static void main(String[] args){
        testdll test = new testdll();
        test.set(10);
        System.out.println(test.get());
    }
}
用javac testdll.java編譯它,會生成testdll.class。再用javah testdll,則會在當前目錄下生成testdll.h文件,這個文件需要被C/C++程序調用來生成所需的庫文件。
二、C/C++中所需要做的工作
對於已生成的.h頭文件,C/C++所需要做的,就是把它的各個方法具體的實現。然後編譯連接成庫文件即可。再把庫文件拷貝到JAVA程序的路徑下面,就可以用JAVA調用C/C++所實現的功能了。
接上例子。我們先看一下testdll.h文件的內容:

#include <jni.h>
#ifndef _Included_testdll
#define _Included_testdll
#ifdef __cplusplus
extern "C" {
#endif
JNIEXPORT jint JNICALL Java_testdll_get(JNIEnv *, jclass);
JNIEXPORT void JNICALL Java_testdll_set(JNIEnv *, jclass, jint);
#ifdef __cplusplus
}
#endif
#endif
在具體實現的時候,我們只關心兩個函數原型

JNIEXPORT jint JNICALL Java_testdll_get (JNIEnv *, jclass);和 JNIEXPORT void JNICALL Java_testdll_set (JNIEnv *, jclass, jint);
這裏JNIEXPORT和JNICALL都是JNI的關鍵字,表示此函數是要被JNI調用的。而jint是以JNI爲中介使JAVA的int類型與本地的int溝通的一種類型,我們可以視而不見,就當做int使用。函數的名稱是JAVA_再加上java程序的package路徑再加函數名組成的。參數中,我們也只需要關心在JAVA程序中存在的參數,至於JNIEnv*和jclass我們一般沒有必要去碰它。

好,下面我們用testdll.cpp文件具體實現這兩個函數:
#include "testdll.h"
int i = 0;
JNIEXPORT jint JNICALL Java_testdll_get (JNIEnv *, jclass)
{
return i;
}
JNIEXPORT void JNICALL Java_testdll_set (JNIEnv *, jclass, jint j)
{
i = j;
}
編譯連接成庫文件,本例是在WINDOWS下做的,生成的是DLL文件。並且名稱要與JAVA中需要調用的一致,這裏就是goodluck.dll
把goodluck.dll拷貝到testdll.class的目錄下,java testdll運行它,就可以觀察到結果了。
例二
一、JAVA中所需要做的工作

在JAVA程序中,首先需要在類中聲明所調用的庫名稱,如下:
public class testdll {
    static
    {
        System.loadLibrary("FANGJIAN");
    }
    public native static String ID_Read();
    public static void main(String[] args){
        testdll test = new testdll();
        String a=test.ID_Read();
        System.out.println(a);
    }
}

用javac testdll.java編譯它,會生成testdll.class。
再用javah testdll,則會在當前目錄下生成testdll.h文件,這個文件需要被C/C++程序調用來生成所需的庫文件。
二、C/C++中所需要做的工作
對於已生成的.h頭文件,C/C++所需要做的,就是把它的各個方法具體的實現,此處就可以在方法體中調用廠家所提供的dll庫文件,來實現調用,並獲得返回值。然後編譯連接成庫文件即可。再把庫文件和廠家提供的庫文件拷貝到JAVA程序的路徑下面,就可以用JAVA調用C/C++所實現的功能了。
需要注意的是必須將jni.h文件和jni_md.h文件考到c編譯器的include文件夾下,這樣才能通過編譯!!!調用了這兩個頭文件裏的聲明……….
接上例子。我們先看一下testdll.h文件的內容:
#include <jni.h>
#ifndef _Included_testdll
#define _Included_testdll
#ifdef __cplusplus
extern "C" {
#endif
JNIEXPORT jstring JNICALL Java_testdll_ID_1Read(JNIEnv *, jclass);//方法名
#ifdef __cplusplus
}
#endif
#endif
在具體實現的時候,我們只關心這個函數原型
JNIEXPORT jstring JNICALL Java_testdll_ID_1Read(JNIEnv *, jclass);
如果是java類文件有包的話,函數的原型的名字會相應加上包名字
如:pakcage com.util; 原型變爲:
JNIEXPORT jstring JNICALL Java_com_util_testdll_ID_1Read(JNIEnv *, jclass);
這裏JNIEXPORT和JNICALL都是JNI的關鍵字,表示此函數是要被JNI調用的。而jint(jstring)是以JNI爲中介使JAVA的int(string)類型與本地的int(string)溝通的一種類型,我們可以視而不見,就當做int(string)使用。函數的名稱是JAVA_再加上java程序的package路徑再加函數名組成的。參數中,我們也只需要關心在JAVA程序中存在的參數,至於JNIEnv*和jclass我們一般沒有必要去碰它。
好,下面我們用testdll.cpp文件具體實現這兩個函數
#include <iostream.h>
#include <windows.h>
#include <string.h>
#include <testdll.h>
typedef HANDLE (_stdcall *COMINT)(unsigned char port);
typedef int (_stdcall *COM_CLOSE)(HANDLE hr);
typedef int (_stdcall *ID_Read)(HANDLE icdev,int tt,unsigned char *_Data);
JNIEXPORT jstring JNICALL Java_testdll_ID_1Read(JNIEnv* env,jclass)
{
HINSTANCE hTest;
HANDLE hr;
COMINT pComInit;
COM_CLOSE pComClose;
ID_Read pRead;
int j;
unsigned char data[16]="put card on it",*A=data;
jstring jstr;
hTest=LoadLibrary("TF_ID.dll");
pComInit=(COMINT)GetProcAddress(hTest,"ComInit");
pComClose=(COM_CLOSE)GetProcAddress(hTest,"ComClose");
pRead=(ID_Read)GetProcAddress(hTest,"ID_Read");
hr=pComInit(1);
j=pRead(hr,3000,data);
char b[16]="put card on it",*B=b;
for(int i=0;i<16;i++)
*(B+i)=*(A+i);
jstr=env->NewStringUTF(b);
if(j==8)
return jstr;
else
return jstr;
pComClose(hr);
FreeLibrary(hTest);
}
其中進行了unsigned char[]到char[]的轉換,並最後付給jstring中!!!

stdcall、cdecl的區別!!!
編譯連接成庫文件,本例是在WINDOWS下做的,生成的是DLL文件。並且名稱要與JAVA中需要調用的一致,這裏就是FANGJIAN.dll

把FANGJIAN.dll和TF_ID.dll(或考TF_ID.dll到system32下)拷貝到testdll.class的目錄下,java testdll運行它,就可以觀察到結果了。


注意:如果是在linux平臺下的JAVA調用動態庫so,需要將生成的so文件名稱在java中聲明的調用動態庫名稱的基礎上添加lib,這個是linux的規則。

比如:LoadLibrary("lkl.so");,則生成的so文件爲:liblkl.so

附錄:

1、如何使用java命令編譯帶包的工程?

http://blog.csdn.net/hong201/article/details/3954603

2、如何使用javah生成C/C++頭文件?

http://www.cnblogs.com/charley_yang/archive/2011/03/13/1983038.html

發佈了31 篇原創文章 · 獲贊 16 · 訪問量 12萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章