JNA介紹
JNA(Java Native Access )提供一組Java工具類用于在運行期動態訪問系統本地庫(native library:如Window的dll)而不需要編寫任何Native/JNI代碼。開發人員只要在一個java接口中描述目標native library的函數與結構,JNA將自動實現Java接口到native function的映射。
優點
JNA可以讓你像調用一般java方法一樣直接調用本地方法。就和直接執行本地方法差不多,而且調用本地方法還不用額外的其他處理或者配置什么的,也不需要多余的引用或者編碼,使用很方便。
JNA描述
JNA類庫使用一個很小的本地類庫sub 動態的調用本地代碼。程序員只需要使用一個特定的java接口描述一下將要調用的本地代碼的方法的結構和一些基本屬性。這樣就省了為了適配多個平臺而大量的配置和編譯代碼。因為調用的都是JNA提供的公用jar 包中的接口。
JNA簡單使用
JNA實現WINDOWS下全局鼠標鍵盤鉤子
package jna_test;
import com.sun.jna.platform.win32.*;
import com.sun.jna.platform.win32.WinUser.*;
import com.sun.jna.platform.win32.WinDef.HMODULE;
import com.sun.jna.platform.win32.WinDef.LRESULT;
import com.sun.jna.platform.win32.WinDef.WPARAM;
import com.sun.jna.platform.win32.WinUser.HHOOK;
import com.sun.jna.platform.win32.WinUser.KBDLLHOOKSTRUCT;
import com.sun.jna.platform.win32.WinUser.LowLevelKeyboardProc;
import com.sun.jna.win32.WinUser.Mouse.LowLevelMouseProc;
import com.sun.jna.win32.WinUser.Mouse.MOUSEHOOKSTRUCT;
public class MouseLLHook {
// 鼠標鉤子函數里判斷按鍵類型的常數
public static final int WM_LBUTTONUP = 514;
public static final int WM_LBUTTONDOWN = 513;
public static final int WM_RBUTTONUP = 517;
public static final int WM_RBUTTONDOWN = 516;
public static final int WM_MOUSEHWHEEL = 526;
public static final int WM_MOUSEWHEEL = 522;
public static final int WM_MOUSEMOVE = 512;
static HHOOK mouseHHK,keyboardHHK;//鼠標、鍵盤鉤子的句柄
static LowLevelMouseProc mouseHook;//鼠標鉤子函數
static LowLevelKeyboardProc keyboardHook;//鍵盤鉤子函數
// 安裝鉤子
static void setHook() {
HMODULE hMod = Kernel32.INSTANCE.GetModuleHandle(null);
mouseHHK = User32.INSTANCE.SetWindowsHookEx(WinUser.WH_MOUSE_LL, mouseHook, hMod, 0);
keyboardHHK = User32.INSTANCE.SetWindowsHookEx(WinUser.WH_KEYBOARD_LL, keyboardHook, hMod, 0);
}
//卸載鉤子
static void unhook() {
User32.INSTANCE.UnhookWindowsHookEx(keyboardHHK);
User32.INSTANCE.UnhookWindowsHookEx(mouseHHK);
}
public static void main(String[] args) {
keyboardHook = new LowLevelKeyboardProc() {
@Override
//該函數參數的意思參考:http://msdn.microsoft.com/en-us/library/windows/desktop/ms644985(v=vs.85).aspx
public LRESULT callback(int nCode, WPARAM wParam, KBDLLHOOKSTRUCT lParam) {
int w = wParam.intValue();
//按下alt鍵時w=.WM_SYSKEYDOWN; 按下其他大部分鍵時w=WinUser.WM_KEYDOWN
if(w==WinUser.WM_KEYDOWN || w==WinUser.WM_SYSKEYDOWN)
System.out.println("key down: vkCode = "+lParam.vkCode);
else if(w==WinUser.WM_KEYUP || w==WinUser.WM_SYSKEYUP)
System.out.println("key up: vkCode = "+lParam.vkCode);
// 如果按下'q'退出程序,'q'的vkCode是81
if(lParam.vkCode==81) {
unhook();
System.err.println("program terminated.");
System.exit(0);
}
return User32.INSTANCE.CallNextHookEx(keyboardHHK, nCode, wParam, lParam.getPointer());
}
};
mouseHook = new LowLevelMouseProc() {
@Override
//該函數參數的意思參考:http://msdn.microsoft.com/en-us/library/windows/desktop/ms644986(v=vs.85).aspx
public LRESULT callback(int nCode, WPARAM wParam, MOUSEHOOKSTRUCT lParam) {
switch (wParam.intValue()) {
case WM_MOUSEMOVE:
System.out.print("mouse moved:");
break;
case WM_LBUTTONDOWN:
System.out.print("mouse left button down:");
break;
case WM_LBUTTONUP:
System.out.print("mouse left button up");
break;
case WM_RBUTTONUP:
System.out.print("mouse right button up:");
break;
case WM_RBUTTONDOWN:
System.out.print("mouse right button down:");
break;
case WM_MOUSEWHEEL:
System.out.print("mouse wheel rotated:");
break;
}
System.out.println("("+lParam.pt.x+","+lParam.pt.y+")");
return User32.INSTANCE.CallNextHookEx(mouseHHK, nCode, wParam, lParam.getPointer());
}
};
System.out.println("press 'q' to quit.");
setHook();
int result;
MSG msg = new MSG();
// 消息循環
// 實際上while循環一次都不執行,這些代碼的作用我理解是讓程序在GetMessage函數這里阻塞,不然程序就結束了。
while ((result = User32.INSTANCE.GetMessage(msg, null, 0, 0)) != 0) {
if (result == -1) {
System.err.println("error in GetMessage");
unhook();
break;
} else {
User32.INSTANCE.TranslateMessage(msg);
User32.INSTANCE.DispatchMessage(msg);
}
}
unhook();
}
}