javajna
⑴ JNA 下面的C++代碼怎麼轉java
Java是通過JNI調用其他語言(包括C++)編譯的本地方法的,而本地方法是以庫文件的形式存放的(在WINDOWS平台上是DLL文件形式,在UNIX機器上是SO文件形式)。
如下是詳細講解:
1、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++程序調用來生成所需的庫文件。
2、C/C++中所需要做的工作
對於已生成的.h頭文件,C/C++所需要做的,就是把它的各個方法具體的實現。然後編譯連接成庫文件即可。再把庫文件拷貝到JAVA程序的路徑下面,就可以用JAVA調用C/C++所實現的功能了。
接上例子。我們先看一下TestDll.h文件的內容:
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class TestDll */
#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文件具體實現這兩個函數:
#include "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運行它,就可以觀察到結果了。
⑵ 請問在java中怎麼使用jna 獲取說有的窗口句柄
import com.sun.jna.platform.win32.User32;
import com.sun.jna.platform.win32.WinDef.HWND;
public class Demo1 {
public static void main(String[] args) {
name();
}
//窗體句柄的截獲
public static void name() {
// HWND hwnd = User32.INSTANCE.FindWindow(null, "列印");
String zh="列印";
HWND hwndDialog = User32.INSTANCE.FindWindow(null,zh);
if(hwndDialog==null) {
System.err.println("沒有找到窗體");
}else {
System.err.println("找到窗體"+hwndDialog);
//HWND root, String className, long timeout, TimeUnit unit,String caption
// List<HWND> hwndButton= Win32Util.findHandleByClassNameJKSD(hwndDialog,"Button", 10, TimeUnit.SECONDS,"列印");
// for (HWND hwnd : hwndButton) {
// System.err.println(hwnd);
// }
⑶ java中jna的問題
1、在VC下創建一個動態鏈接庫項目testJNA
2、在頭文件里聲明函數
C代碼 收藏代碼
extern "C" _declspec(dllexport) int add(int first, int second);
紅色字體部分是必須的,包括定義結構體時也需要。應該是說此函數是發布的。
3、在源碼里實現函數
C代碼 收藏代碼
int add(int first, int second) {
printf("(c) test jna : %d + %d = %d", first, second, first + second);
return first + second;
}
4、生成dll文件
5、定義一個表示鏈接庫的介面
介面TestJnaLib繼承自com.sun.jna.Library,此介面有一個實例
Java代碼 收藏代碼
TestJnaLib INSTANCE = (TestJnaLib)Native.loadLibrary("testJNA.dll", TestJnaLib.class);
此實例由jna通過反射自動生成。
6、定義對應dll里的方法
Java代碼 收藏代碼
int add(int first, int second);
7、調用本地方法
Java代碼 收藏代碼
TestJnaLib.INSTANCE.add(3, 5);
Jna回調Java方法:
1、在C語言部分定義帶回調函數的函數
C代碼 收藏代碼
extern "C" _declspec(dllexport) void methodWithCallback(int (*fp)(int left, int right), int left, int right);
紅色加粗部分是函數指針。
2、Java部分定義一個回調介面
必須繼承自com.sun.jna.Callback介面
Java代碼 收藏代碼
public interface FunCallBack extends Callback {
int invoke(int left, int right);
}
Invoke方法里的參數順序與C函數的對應
3、定義回調介面的實現
Java代碼 收藏代碼
public class CallbackFunImpl implements FunCallBack {
@Override
public int invoke(int left, int right) {
System.out.printf("in java :%d + %d = %d\n", left, right, left + right);
return left + right;
}
}
4、在表示鏈接庫實現的介面里定義要回調的本地函數
Java代碼 收藏代碼
void methodWithCallback(Callback callback, int left, int right);
本地函數的函數指針用Callback 介面替代。
5、調用帶函數指針的本地函數
Java代碼 收藏代碼
TestJnaLib.INSTANCE.methodWithCallback(new CallbackFunImpl(), 4, 6)