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。
推薦閱讀:
- Java中的Native方法
- Mac下用Java調用c/c++的思路詳解
- Java調用C++程序的實現方式
- Java中的Native關鍵字講解
- windows下vscode+vs2019開發JNI的示例