JNA 是您想要使用的。在 Java 中更难的原因是 Java 被构建为与操作系统无关,每当您尝试做一些过于接近操作系统内部工作的事情时,您必须离开核心 Java 并使用平台相关的解决方案一种或另一种。
例如,这是获取当前使用 JNA 运行的窗口名称的一种方法,但我还没有弄清楚如何将其限制为仅顶级窗口:
import java.util.ArrayList;
import java.util.List;
import com.sun.jna.Native;
import com.sun.jna.Pointer;
import com.sun.jna.win32.StdCallLibrary;
public class EnumAllWindowNames {
static interface User32 extends StdCallLibrary {
User32 INSTANCE = (User32) Native.loadLibrary("user32", User32.class);
interface WNDENUMPROC extends StdCallCallback {
boolean callback(Pointer hWnd, Pointer arg);
}
boolean EnumWindows(WNDENUMPROC lpEnumFunc, Pointer userData);
int GetWindowTextA(Pointer hWnd, byte[] lpString, int nMaxCount);
Pointer GetWindow(Pointer hWnd, int uCmd);
Pointer GetParent(Pointer hWnd);
Pointer GetAncestor(Pointer hWnd, int gaFlags);
}
public static String[] ignoreArray = { "Default", "MSCTFIME", "???????", "Msg",
"SysFader" };
public static boolean ignore(String text) {
for (String ignoreText : ignoreArray) {
if (ignoreText.contains(text)) {
return true;
}
}
return false;
}
public static List<String> getAllWindowNames() {
final List<String> windowNames = new ArrayList<String>();
final User32 user32 = User32.INSTANCE;
user32.EnumWindows(new User32.WNDENUMPROC() {
@Override
public boolean callback(Pointer hWnd, Pointer arg) {
byte[] windowText = new byte[512];
user32.GetWindowTextA(hWnd, windowText, 512);
String wText = Native.toString(windowText).trim();
if (!ignore(wText)) {
if (user32.GetParent(hWnd) == null) {
windowNames.add(wText);
}
}
return true;
}
}, null);
return windowNames;
}
public static void main(String[] args) {
List<String> winNameList = getAllWindowNames();
for (String winName : winNameList) {
System.out.println(winName);
}
}
}
这需要使用两个 jna jar 文件:platform.jar 和 jna.jar。