1. 程式人生 > 其它 >java呼叫dll(native方法的實現)

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 裡。

【當你用心寫完每一篇部落格之後,你會發現它比你用程式碼實現功能更有成就感!】