java呼叫dll(native方法的實現)
java 中有許多native 方法,下面簡單研究下native 方法的實現以及在java 中呼叫native 方法。
下面以簡單的操作加減乘除實現
1. 新建java 類
原始碼如下:
package com.zd.bx; public class Operation { public native int add(int a, int b); }
2. javah 生成 .h 標頭檔案
.h 檔案是c++的標頭檔案
E:\ideaspace\mvnpro\target\classes>javah com.zd.bx.Operation
最後會在當前目錄生成:com_zd_bx_Operation.h, 內容如下:
/* DO NOT EDIT THIS FILE - it is machine generated */ #include <jni.h> /* Header for class com_zd_bx_Operation */ #ifndef _Included_com_zd_bx_Operation #define _Included_com_zd_bx_Operation #ifdef __cplusplus extern "C" { #endif /* * Class: com_zd_bx_Operation * Method: add * Signature: (II)I*/ JNIEXPORT jint JNICALL Java_com_zd_bx_Operation_add (JNIEnv *, jobject, jint, jint); #ifdef __cplusplus } #endif #endif
3. 用visual studio 生成dll 連結庫
1. 新建專案(選擇c++ -> 動態連結庫)
然後輸入名稱:
2. 生成的目錄結構如下:
3. 將上面生成的com_zd_bx_Operation.h 拷貝到專案目錄下
(1) 拷貝到:E:\visualstudio\namespace\OperationDLL\OperationDLL, 就是和dllmain.cpp 同級目錄
(2) 然後點選標頭檔案, 選擇新增現有項, 選擇上面新增進去的com_zd_bx_Operation.h 檔案
4. 開啟com_zd_bx_Operation.h 會報錯找不到jni.h
5. jni.h 以及相關實現是jdk 提供的, 所以需要引入%jdk%/include, 以及%jdk%/include/win32 目錄作為附加包含目錄
(1) 選擇專案-》 屬性 -》c++ -》常規-》附加包含目錄
(2) 選中 %java%/include 和 %java%/include/win32 目錄
(3) 應用之後再次開啟com_zd_bx_Operation.h 可以看到不會編譯報錯
6. 編輯dllmain.cpp, 將容修改為如下:
// dllmain.cpp : 定義 DLL 應用程式的入口點。 #include "pch.h" #include "com_zd_bx_operation.h" JNIEXPORT jint JNICALL Java_com_zd_bx_Operation_add (JNIEnv* env, jobject obj, jint a, jint b) { return a + b; }
7. 點選導航欄 生成 -》 生成解決方案
控制檯顯示如下:(會顯示dll 的生成位置)
已啟動生成… 1>------ 已啟動生成: 專案: OperationDLL, 配置: Debug x64 ------ 1>dllmain.cpp 1> 正在建立庫 E:\visualstudio\namespace\OperationDLL\x64\Debug\OperationDLL.lib 和物件 E:\visualstudio\namespace\OperationDLL\x64\Debug\OperationDLL.exp 1>OperationDLL.vcxproj -> E:\visualstudio\namespace\OperationDLL\x64\Debug\OperationDLL.dll ========== 生成: 成功 1 個,失敗 0 個,最新 0 個,跳過 0 個 ==========
也就是生成了所需要的dll 庫
4. java 呼叫dll
1. 將上面的dll 拷貝到 java工程目錄下:
2. 編寫測試程式碼:
package com.zd.bx; public class PlainTest { static { System.loadLibrary("OperationDLL"); } public static void main(String[] args) { System.out.println(new Operation().add(1, 3)); } }
結果:
4
5. 改進
1. c++ 的cout 也可以輸出到控制檯, 比如修改dllmain.cpp
// dllmain.cpp : 定義 DLL 應用程式的入口點。 #include "pch.h" #include "com_zd_bx_operation.h" #include <iostream> using namespace std; JNIEXPORT jint JNICALL Java_com_zd_bx_Operation_add (JNIEnv* env, jobject obj, jint a, jint b) { int version = env->GetVersion(); cout << "env " << env << endl; cout << "env->GetVersion() " << version << endl; cout << "obj " << obj << endl; return a + b; }
上面程式碼呼叫 env.GetVersion() 方法。 然後列印相關物件記憶體地址。 關於env 和 obj 有哪些方法以及屬性可以Ctrl + 滑鼠左鍵進去檢視,類似於java 檢視類方法。
重新生成dll 後測試如下:
env 000002625ABCFA00 env->GetVersion() 65544 obj 000000EDD66FF2D0 4
2. 可以將c++實現和主類進行隔離。
(1) VS中選擇標頭檔案然後新增 Operation.h 標頭檔案
內容如下:
#pragma once int add(int a, int b);
(2) 原始檔選擇新增信件項選擇 cpp 檔案
內容如下:
#include "pch.h" #include <stdio.h>; #include "Operation.h"; using namespace std; int add(int a, int b) { printf("cpp print a: %i b: %i", a, b); return a + b; }
(3) 修改dllmain.cpp
// dllmain.cpp : 定義 DLL 應用程式的入口點。 #include "pch.h" #include "com_zd_bx_operation.h" #include "Operation.h" JNIEXPORT jint JNICALL Java_com_zd_bx_Operation_add (JNIEnv* env, jobject obj, jint a, jint b) { return add(a, b); }
(4) 最終目錄結構如下:
(5) 重新生成dll 後測試結果如下:
總結:
.h 檔案我理解類似於java 的介面, 只給出定義。 具體的cpp 檔案引入之後可以給出方法的實現。 然後別的模組引入相關標頭檔案即可(標頭檔案的方法只能有一個cpp 中有,否則會報錯)。 我們用javah 生成的也是.h 標頭檔案, 所以我們需要做的就是生成實現的方法, 然後匯出到dll 裡。
【當你用心寫完每一篇部落格之後,你會發現它比你用程式碼實現功能更有成就感!】