『壹』 java調用dll如何傳入指針invoke後如何獲取指針的值
假設我有一個叫test.dll的動態庫,其中有一個函數如下:
extern "C" int add(int* a,int* b,int* c,int d,char* PicPath)
{
FILE *fp;
fp=fopen(PicPath,"wt");
for(int i=0;i
{
c[i]=a[i]+b[i];
fprintf(fp,"%d\n",c[i]);
}
fclose(fp);
return 1;
}
該函數的功能是a,b兩個數組的值相加,和存放的數組c中,在將和寫入到指定的文件,數組的維數
有傳入的參數決定。
問題提出:
1.Java中如何調用指針類型的參數;
2.整形的參數可以直接傳入嗎?;
3.PicPath為何是指針型的,怎麼不是CString類型的;
4.如何獲取輸入型參數的返回值;
Java調用動態鏈接庫的代碼如下:
JNative t= new JNative("test.dll", "add"); //載入test.dll文件,並調用add方法;
t.setRetVal(Type.INT); // 設置調用後的返回類型
// Java中構造指針,並為指針分配空間
Pointer a = new Pointer(MemoryBlockFactory.createMemoryBlock(4 * 10));
Pointer b = new Pointer(MemoryBlockFactory.createMemoryBlock(4 * 10));
Pointer c = new Pointer(MemoryBlockFactory.createMemoryBlock(4 * 10));
Pointer strP= createPointer(100);
strP.setStringAt(0, "C:\\1.txt");
int d=10;
// 為數組a和b賦值
for (int i = 0; i
a.setIntAt(4 * i, i);
b.setIntAt(4 * i, i);
}
// 設置參數:
t.setParameter(0, a);
t.setParameter(1, b);
t.setParameter(2, c);
t.setParameter(3, d);
t.setParameter(4, strP);
t.invoke(); // 調用函數並執行
// 輸出和的值,注意:一個整形值需要用4個位元組存儲,所以j前面需要乘以4
for (int j = 0; j
System.out.println(c.getAsInt(4 * j));
}
// 釋放內存
a.dispose();
b.dispose();
c.dispose();
好了,這樣一個完整的調用例子就結束了,我只是寫出了程序中的關鍵的代碼,具體的還需要寫一
個類,並且繼承一個基類,在復寫基類中的方法,這些都比較簡單,大家可以參考一下網上的其他
例子。
問題解答:
1.Java中調用指針類型的參數:需要創建Pointer變數,並為其分配空間
2.整形的數據可以直接傳入;
3.JNative對CString類型支持不是很良好,調用CString的方法時,經常會報一些異常,所以我把路徑改成
的字元指針;
4.輸入型參數作為輸出:可以參見以上的例子,數組c是如何設置和輸出的。
『貳』 怎樣用JAVA調用DLL
1. 編寫範例文檔
public class TestNative
{
private native static int Max(int a,int b);
public static void main(String[] args)
{
System.out.println(Max(4,5));
}
static
{
System.loadLibrary("VCdll");
}
}
其中
LoadLibrary中的DLL文件名稱可以隨便定.只要做好的動態鏈接庫改名成這個字元串就可以了.而native函數的來源就是該動態鏈接庫.
2. 做好該步准備以後,用javac TestNative.java生成TestNative.class. 檢驗語法.
3. 語法檢查通過以後,用javah TestNative生成TestNative.h的C++頭文件.
4. 用VC++6.0創建dll文件:
<1>msdev->new->project->win32 Dynamic-Link Library
<2>Enter project name (e.g. VCdll)
<3>choose "An empty DLL project"->finish
<4>將3生成的TestNative.h和jdk下的jni.h和jni_md.h加入到工程中去
<5>Create a c++ file, implements the native method. as follows:
//TestNative.cpp
#include "TestNative.h"
JNIEXPORT jint JNICALL Java_TestNative_Max
(JNIEnv * a, jclass b, jint x, jint y)
{
return x > y ? x : y;
}
備注:
A 其中的jni.h通常在 jdk\include\ 可以先拷貝到VC安裝目錄下的include目錄下.一勞永逸,以後可以方便使用JNI技術.
B 全部的數據類型需要考慮Java和VC的介面問題.用jint jstring 等.
C 函數名稱和自動生成的.h文件中規定的一致.命名規則是: Java_使用類_方法名
6. 全編譯一下VC++工程.
如果沒有出錯,則此時生成動態鏈接庫成功.
把動態鏈接庫VCdll.dll拷貝到TestNative.java所在的目錄上.
7.運行java TestNative. OK :-)
『叄』 如何在java中調用.dll文件,詳細點帶上代碼這些辣
首先下截JNative組件
jnative.sourceforge.net/ 到這里下載JNative開源項目,我下載的是1.3.2
解壓JNative-<st1:chsdate isrocdate="False" islunardate="False" day="30"
month="12" year="1899">1.3.2</st1:chsdate>.zip
獲得三個文件,分別是:JNativeCpp.dll,libJNativeCpp.so,JNative.jar 。
JNativeCpp.dll
Windows下用的,拷貝到windows / system32目錄下;
libJNativeCpp.so
Linux下的,拷貝到系統目錄下;
JNative.jar這是一個擴展包,導入工程LIB中或將其拷貝到jdk\jre\lib\ext
下,系統會自動載入。
•使用說明
我的項目將使用JNative組件調用一個測試應用伺服器狀態的TestAppSvr.dll文件,Dll文件中包含一個TestConnect()方法,返回一個整形的結果(1或0)
首先配置好JNative組件的windows環境:
將Native要用到JNativeCpp.dll放在系統盤的\WINDOWS\system32下
將JNative.jar導入工程中,新建一個調用類:
java 代碼
復制代碼
代碼如下:
package com.tvjody;
import
java.io.File;
import java.io.FileOutputStream;
import
java.io.IOException;
import java.io.InputStream;
import
org.xvolks.jnative.JNative;
import org.xvolks.jnative.Type;
import
org.xvolks.jnative.exceptions.NativeException;
public class
AppSvrTestConnect {
public AppSvrTestConnect() {
}
/**
* 測試應用伺服器連接狀態
*
* TestConnect
* @param ip 應用伺服器IP
* @param port 埠
* @param
intrcpt 是否採用數據壓縮方式 1 :true 0:false
* @return int 1 :成功 0:失敗
* @throws NativeException
* @throws IllegalAccessException
*/
private static final int TestConnect(String ip, int port, int
intrcpt)throws NativeException, IllegalAccessException {
JNative n
= null;
try {
n = new
JNative("TestAppSvr.dll", "TestConnect");
n.setRetVal(Type.INT);
int i = 0;
n.setParameter(i++, Type.STRING, ip);
n.setParameter(i++,
Type.INT, "" + port);
n.setParameter(i++, Type.INT, "" +
intrcpt);
n.invoke();
return
Integer.parseInt(n.getRetVal());
} finally {
if
(n != null)
n.dispose();
}
}
/**
* 指定Dll文件路徑,動態載入本地鏈接庫,測試應用伺服器連接狀態
* setDllPath
* @param path Dll文件的路徑,不包含DLL名稱 例如:windows - d:\test\test\ unix -
root/test/test/
* @param ip 應用伺服器IP
* @param port 埠
* @param intrcpt 是否採用數據壓縮方式 1 :true 0:false
* @return int 1
:成功 0:失敗
* @throws NativeException
* @throws
IllegalAccessException
*/
public static final int
TestConnectFromDllPath(String path,String ip, int port, int intrcpt) throws
NativeException, IllegalAccessException{
path +=
"TestAppSvr.dll";
System.load(path);
return
TestConnect(ip,port,intrcpt);
}
/**
*
Dll文件放在JRE\bin目錄下面,ClassLoader就能通過System.loadLibrary()動態載入本地鏈接庫
*
TestConnectFromDllPath
* @param ip 應用伺服器IP
* @param port 埠
* @param intrcpt 是否採用數據壓縮方式 1 :true 0:false
* @return int 1
:成功 0:失敗
* @throws NativeException
* @throws
IllegalAccessException
*/
public static final int
TestConnectFromDllPath(String ip, int port, int intrcpt) throws NativeException,
IllegalAccessException{
System.loadLibrary("TestAppSvr");
return TestConnect(ip,port,intrcpt);
}
}
這個類實現了一個靜態私有方法,用來調用Dll文件中的方法返回結果
private static final int TestConnect(String ip, int port, int intrcpt)
兩個靜態公共方法,分兩種方式裝載DLL文件
public static final int TestConnectFromDllPath(String path,String ip, int
port, int intrcpt) //通過DLL文件的路徑
public static final int
TestConnectFromDllPath(String ip, int port, int intrcpt)
//通過ClassLoader
然後新建一個類,調用AppSvrTestConnect.java,實現方法一調用,我是將TestAppSvr.dll文件與Demo.java放在一個目錄下
,所以得到Demo.java的路徑後就可以得到TestAppSvr.dll的路徑,調用AppSvrTestConnect.TestConnectFromDllPath()方法後就能返回正確的信息.方法二是已經將TestAppSvr.dll放在了Jre\bin目錄下,在JVM的Classloader的時候會自動載入,然後通過System.loadLibrary("TestAppSvr")就可以裝配DLL文件.
java 代碼
復制代碼
代碼如下:
public class Demo {
public int
getInfo() throws NativeException, IllegalAccessException{
String path=getClass().getResource(File.separator).getPath();
path = path.substring(1,path.length());
System.out.println(path); //得到DLL文件的路徑
String ip =
"192.168.0.48"; //伺服器IP
int port = 221; //埠
int intrcpt = 1; //數據壓縮方式傳送,1為採用;0為不採用
//方法1 傳入Dll文件的路徑
//int info =
AppSvrTestConnect.TestConnectFromDllPath(path, ip, port, intrcpt);
//方法2 Dll文件已經放在JRE\bin目錄下面
int info =
AppSvrTestConnect.TestConnectFromDllPath(ip, port, intrcpt);
//1為成功,0為失敗
if (info == 1)
System.out.println("應用伺服器可用。");
else
System.out.println("應用伺服器不可用,請檢查IP地址和埠是否正確。");
return info;
}
System.loadLibrary():裝載Windows\System32下或jre\bin或Tomcat\bin目錄下的本地鏈接庫
System.load():根據具體的目錄來加截本地鏈接庫,必須是絕對路徑
•備注
上面的示例工程,因為是例子,所以沒有大多的設計,只是實現了裝載DLL文件,調用DLL文件方法,返回信息.
注意JVM只允許一個默認的ClassLoader來load native library,同時並不提供專門的API來unload一個loaded
native library,所以在項目調試的時候,獨立啟動Web Server.
『肆』 java 如何調用dll
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文件的內容:
#i nclude <jni.h>
#ifndef _Included_testdll
#define _Included_testdll
#ifdef __cplusplus
extern "C" {
#endif
JNIEXPORT jint JNICALL Java_testdll_get (JNIEnv *, jclass);
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文件具體實現這兩個函數:
#i nclude "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運行它,就可以觀察到結果了。