<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    posts - 4,  comments - 9,  trackbacks - 0

    ????????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語言方法
    (加入日期:2002-3-18 點擊數:9803)
    【對此文發表評論】 【編程愛好者論壇】 【保存文章至硬盤】? 【打印文章】
    ?
    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文件的內容:

    /* DO NOT EDIT THIS FILE - it is machine generated */

    #include <jni.h>

    /* Header for class testdll */

    #ifndef _Included_testdll

    #define _Included_testdll

    #ifdef __cplusplus

    extern "C" {

    #endif

    /*

    * Class: testdll

    * Method: get

    * Signature: ()I

    */

    JNIEXPORT jint JNICALL Java_testdll_get

    (JNIEnv *, jclass);

    /*

    * Class: testdll

    * Method: set

    * Signature: (I)V

    */

    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文件的內容:

    /* DO NOT EDIT THIS FILE - it is machine generated */
    #include <jni.h>
    /* Header for class testdll */

    #ifndef _Included_testdll
    #define _Included_testdll
    #ifdef __cplusplus
    extern "C" {
    #endif
    /*
    ?* Class:???? testdll
    ?* Method:??? ID_Read
    ?* Signature: ()Ljava/lang/String;
    ?*/

    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運行它,就可以觀察到結果了。
    posted on 2006-11-01 13:05 凌宇 閱讀(324) 評論(0)  編輯  收藏

    只有注冊用戶登錄后才能發表評論。


    網站導航:
     
    <2025年5月>
    27282930123
    45678910
    11121314151617
    18192021222324
    25262728293031
    1234567

    常用鏈接

    留言簿(3)

    隨筆檔案(3)

    文章檔案(14)

    相冊

    收藏夾

    Java

    最新隨筆

    搜索

    •  

    最新評論

    閱讀排行榜

    評論排行榜

    主站蜘蛛池模板: 国产成人高清亚洲一区91| 亚洲熟妇AV一区二区三区浪潮| 四虎永久精品免费观看| 国产精品成人免费综合| 亚洲综合在线另类色区奇米| 国产亚洲精品不卡在线| 亚洲欧洲日韩在线电影| 亚洲乱亚洲乱妇无码| 青柠影视在线观看免费| 国产啪精品视频网免费| 国产zzjjzzjj视频全免费 | 任你躁在线精品免费| 国产午夜亚洲精品国产成人小说| 色老头综合免费视频| 午夜视频免费在线观看| 国产一精品一AV一免费孕妇| 亚洲午夜国产片在线观看| 亚洲日韩国产精品无码av| 久久九九兔免费精品6| 亚洲人成影院在线观看| 国产免费伦精品一区二区三区| 国产卡二卡三卡四卡免费网址 | 国产三级在线观看免费| 亚洲国产成人久久一区二区三区| a毛片免费全部播放完整成| 国产精品国产免费无码专区不卡| 色欲aⅴ亚洲情无码AV| 皇色在线视频免费网站| 亚洲视频在线视频| 一个人看的www免费高清| 欧美日韩国产免费一区二区三区| 亚洲色少妇熟女11p| 亚洲国产精品日韩| 亚洲综合欧美色五月俺也去| 波多野结衣视频在线免费观看| 亚洲成人激情小说| h视频在线免费看| 亚洲AV永久无码天堂影院| 嫩草成人永久免费观看| 国产亚洲精品高清在线| 亚洲综合免费视频|