Java調用dll文件的實現解析

Java調用dll文件

近期根據C++做瞭一個圖片質量檢測的項目,目前需要在在java中進行調用,所以先在C++上生成dll文件,然後基於java調用dll文件實現功能。

環境

C++:VS2017(之前配置opencv真是要瞭老命)

java:idea2020+jdk1.8。

註意:jdk安裝的時候小心點,path路徑容易點編輯,千萬別新建,會覆蓋的。

接下來進入正文

1. 創建Java項目,假設定義HelloWorld函數,其中“winproject1”是等會要調用的dll文件,現在進入cmd並cd到當前目錄下面,然後javah -jni HelloWorld.HelloWorld,這樣會在目錄下面生成一個HelloWorld_HelloWorld.h。

等會需要將這個頭文件移到之前安裝jdk目錄裡的include下面,如:D:%你的路徑%\Java\include。

2. 打開vs2017,創建一下新的控制臺項目,然後需要配置項目包含目錄的路徑,將下面兩個路徑加進去,保險點還可以在附加目錄裡面加上這些路徑。

vs2017中編寫上面頭文件中的代碼:

JNIEXPORT void JNICALL Java_HelloWorld_HelloWorld_sayHello (JNIEnv *, jobject, jstring, jstring, jstring, jstring) { cout<<"hello world!"<<endl; }

**註:**其中#include "single_check.h"就是我定義檢測函數的頭文件,在下面的函數中可以調用自己定義的函數,從而讓java執行裡面的內容,可以調用函數。

3.生成第一步中提到的winproject1.dll文件,直接點擊生成-生成解決方案,便可以在項目路徑(%項目名%\x64\Debug\winproject1.dll)中找到dll文件,並且復制到jdk安裝路徑的bin文件夾下(D:%你的路徑%\Java\bin\)

4.最後一步最簡單啦,在java裡面直接run就行啦。最後看一下結果!完美!

這是windows下用java調用dll文件,接下來要實現跨平臺調用,得在linux下用java可以實現改功能。頭禿啊

Java調用dll文件幾種常見方式

Java調用動態庫需要關註的問題

1.如何裝載DLL文件,以及如何定位所要使用的方法;

2.數據類型如何對應;

3.如何給使用的方法傳遞參數;

4.如何獲取返回的值。

一.數據類型對應關系

Java Type C Type
boolean int
byte char
char wchar_t
short short
double double
float float
String char*

二.Jnative調用dll

Jnative是對JNI技術進行瞭封裝,更加方便的讓java去調用DLL。

1. 下載Jnative庫,其中包含JNative.jar, JNativeCPP.dll, JNativeCPP.so這三個包。 JNative.jar是需要導入到Java工程的lib下, JNativeCPP.dll文件放在jdk安裝目錄下,或者是user\System32目錄下,或者項目根目錄下。

2. 將需要調用的dll動態鏈接庫放在SYSTEM32文件夾下,或者是項目根目錄下,否則會出現找不到dll文件的錯誤。

3. 加載DLL庫 : System.loadLibrary("TranferEth"); // TransferEth為需要調用的DLL文件,隻需要使用DLL文件的文件名即可。

4. 調用DLL入口函數 : JNative jnt = new JNative("TransferEth.dll", "Transfer_Ethernet"); // 參數1為需要調用的DLL文件, 參數2為需要調用的方法。

5. 設置返回參數類型 : jnt.setVal(Type.INT);

6. 設置傳入參數 : jnt.setParameter(0, "TransferScale.ini");

7. 執行調用 : jnt.invoke();

8. 釋放資源 : jnt.dispose();

三.JNA調用dll  

JJNA中,它提供瞭一個動態的C語言編寫的轉發器,可以自動實現Java和C的數據類型映射。你不再需要編寫C動態鏈接庫。  

1.在java項目中引入jna.jar。 (當前引用的版本為3.4.0 , 之前的版本未提供釋放動態鏈接庫資源的方法)

2.定義調用接口, 接口方法與需要調用的DLL提供的外部函數一致。

例如:DLL文件中提供入口函數:

extern "C" __declspec(dllexport) UINT __stdcall Transfer_Ethernet_EX(char *cTransScale , int nKey);
extern "C" __declspec(dllexport) UINT __stdcall Transfer_Ethernet(char *cTransScale );

則定義的接口類中如下:  (數據類型對應關系如上)

public interface CallMTScaleLibrary extends Library {  
    public int Transfer_Ethernet_EX(String filename, int key);     
    public int Transfer_Ethernet(String filename); 
}

3.加載DLL文件

String dir = path + "dllName";  // dll文件的路徑,可以省略後綴名,dll和so後綴都可以加載
CallMTScaleLibrary scaleLibrary =  (CallMTScaleLibrary) Native.loadLibrary(dir, CallMTScaleLibrary.class);

4.執行調用

int result = scaleLibrary.Transfer_Ethernet(path + "TransferScale.ini");

5.釋放資源

一次調用完成後需要釋放掉資源,以便後續重復調用該資源。通過查看jna的源碼Native類中有私有方法dispose()可以主動釋放掉資源,可以通過反射的方式去調用該方法,也可以重寫該方法,從而實現釋放。

private static void  dispose(){
           NativeLibrary.disposeAll();         
           nativeLibraryPath = null;
}

註意問題:

1.文件路徑:DLL文件最好放在項目的根目錄下,或者是system32文件夾下,必須在環境變量配置的path中。                 

2.循環調用:當循環調用同一個DLL文件時,必須要釋放掉上一次的資源,否則會被占用端口(每次調用都會在線程裡進行一次網絡通訊)。 

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: