有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

用JavaJNA编写的密钥侦听器。无法停止线程

我使用以下代码来侦听全局关键事件:

Win32HookManager。爪哇

import com.sun.jna.platform.win32.Kernel32;
import com.sun.jna.platform.win32.User32;
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;
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.platform.win32.WinUser.MSG;


import java.awt.event.KeyEvent;

public class Win32HookManager {
    private static HHOOK keyboardHook;

    public static boolean installKeyboardHook(final NativeKeyboardListener listener) {
        final User32  lib  = User32.INSTANCE;
        final HMODULE hMod = Kernel32.INSTANCE.GetModuleHandle(null);

        final LowLevelKeyboardProc keyboardHookProc = new LowLevelKeyboardProc() {
            @Override
            public LRESULT callback(int nCode, WPARAM wParam, KBDLLHOOKSTRUCT info) {
                NativeKeyboardEvent ev = null;
                long                ti = System.currentTimeMillis();
                boolean             nh = true;

                if (nCode >= 0) {
                    switch (wParam.intValue()) {
                        case WinUser.WM_KEYDOWN:
                        case WinUser.WM_SYSKEYDOWN:
                            ev = new NativeKeyboardEvent(KeyEvent.KEY_PRESSED, ti, 0, info.vkCode);
                            nh = listener.keyPressed(ev);
                            break;

                        case WinUser.WM_KEYUP:
                        case WinUser.WM_SYSKEYUP:
                            ev = new NativeKeyboardEvent(KeyEvent.KEY_RELEASED, ti, 0, info.vkCode);
                            nh = listener.keyReleased(ev);
                            break;
                    }
                }

                if(nh) {
                    return lib.CallNextHookEx(keyboardHook, nCode, wParam, info.getPointer());
                }
                return new LRESULT(1);
            }
        };

        new Thread() {
            @Override
            public void run() {
                keyboardHook = lib.SetWindowsHookEx(WinUser.WH_KEYBOARD_LL, keyboardHookProc, hMod, 0);
                msgLoop();
                lib.UnhookWindowsHookEx(keyboardHook);
            }
        }.start();

        return keyboardHook != null;
    }

    public static boolean uninstallKeyboardHook() {
        if(keyboardHook != null) {
            return User32.INSTANCE.UnhookWindowsHookEx(keyboardHook);
        }

        return false;
    }

    private static void msgLoop()
    {
        final User32 lib = User32.INSTANCE;

        int result;
        MSG msg = new MSG();
        while ((result = lib.GetMessage(msg, null, 0, 0)) != 0) {
            if (result == -1) {
                System.err.println("error in get message");
                break;
            }
            else {
                System.err.println("got message");
                lib.TranslateMessage(msg);
                lib.DispatchMessage(msg);
            }
        }
    }
}

本机键盘侦听器

public interface NativeKeyboardListener {
    public boolean keyPressed (NativeKeyboardEvent e);
    public boolean keyReleased(NativeKeyboardEvent e);
}

本机键盘事件

public class NativeKeyboardEvent {
    private int  id;
    private int  keyCode;

    public NativeKeyboardEvent(int id, long when, int modifiers, int keyCode) {
        this.id        = id;
        this.keyCode   = keyCode;
    }

    public int getId() {
        return id;
    }

    public int getKeyCode() {
        return keyCode;
    }
}

不幸的是,它并没有像我预期的那样工作,即它检测何时按下/释放一个键,但它无法完成由installKeyboardHook()方法启动的线程,因为msgLoop方法中有GetMessage()。是的,我可以停止收听关键事件,但我无法停止线程。然而,这段代码中似乎需要GetMessage()。你看到这个问题的解决方法了吗

谢谢


共 (3) 个答案

  1. # 1 楼答案

    正如@SLaks所建议的,您需要某种标志来指示消息循环是否应该继续运行。然后可以使用^{}JNA Doc),它与GetMessage一样,从队列中检索消息,但不是阻塞操作。然后,您的消息循环应更改为以下内容:

    while (!shouldQuit) {
        while ((result = lib.PeekMessage(msg, null, 0, 0, 1)) != 0) {
            // ...
        }
    }
    
  2. # 2 楼答案

    不要在邮件处理线程中安装/卸载

    只有GetMessage处理(和退出检查)需要在附加的Thread中发生(顺便说一句,它应该是守护进程线程)。应在其他地方安装和拆卸挂钩

    查看JNA中提供的keyboard hook以获得正确过程的示例

  3. # 3 楼答案

    您应该设置一个private boolean shouldQuit标志,如果该标志为真,则中断消息循环

    然后,要停止线程,只需将该标志设置为true