如何检查用户是否按下键盘按键?

35
在Java中,我有一个程序需要持续检查用户是否按下了某个键。伪代码如下:
if (isPressing("w")) {
   // do somthing
}

2
在Java中,但使用特定的GUI框架(Swing?)? - mael
4个回答

51

在Java中,你不需要检查按键是否被按下,而是要监听KeyEvent事件。 实现你的目标的正确方式是注册一个KeyEventDispatcher,并实现它来维护所需按键的状态:

import java.awt.KeyEventDispatcher;
import java.awt.KeyboardFocusManager;
import java.awt.event.KeyEvent;

public class IsKeyPressed {
    private static volatile boolean wPressed = false;
    public static boolean isWPressed() {
        synchronized (IsKeyPressed.class) {
            return wPressed;
        }
    }

    public static void main(String[] args) {
        KeyboardFocusManager.getCurrentKeyboardFocusManager().addKeyEventDispatcher(new KeyEventDispatcher() {

            @Override
            public boolean dispatchKeyEvent(KeyEvent ke) {
                synchronized (IsKeyPressed.class) {
                    switch (ke.getID()) {
                    case KeyEvent.KEY_PRESSED:
                        if (ke.getKeyCode() == KeyEvent.VK_W) {
                            wPressed = true;
                        }
                        break;

                    case KeyEvent.KEY_RELEASED:
                        if (ke.getKeyCode() == KeyEvent.VK_W) {
                            wPressed = false;
                        }
                        break;
                    }
                    return false;
                }
            }
        });
    }
}

那么您总是可以使用:

if (IsKeyPressed.isWPressed()) {
    // do your thing.
}

当然,你可以使用相同的方法来实现isPressing("<some key>"),其中包含键和其状态的映射封装在IsKeyPressed中。


1
抱歉,我在没有检查的情况下更新了我的代码。请查看我的更新答案 - 它使用类对象而不是实例。再试一次(只需将IsKeyPressed.this替换为IsKeyPressed.class)。顺便说一句,在那种情况下,synchronized块并不是真正必要的。 - Elist
7
我在测试这个程序,但好像没有起作用:{。我有一个计时器循环,检测是否按下了W键(IsKeyPressed.isWPressed()),但即使我按住或按下W键,它也从未返回True。 - Saucymeatman
@Saucymeatman,你解决问题了吗?如果你一直在等待isWPressed(),那么可能会出现缓存问题。使用volatile boolean可能可以解决这个问题。 - Elist
2
如果您想测试很多键,此解决方案可以轻松扩展以使用“HashMap<Integer, Boolean>”。整数表示“KeyCode”,布尔值则不言自明。 - mindoverflow
@mindoverflow,我会使用Set<Integer>代替HashMap,当按下键时将其添加到集合中,松开键时将其移除。Set<>比HashMap要小得多,因为它只需要容纳同时按下的最大键数(假设为10-20),而不是键盘上可以按下的所有键数(几十个)。 - javmarina

5

试试这个:

import java.awt.event.KeyAdapter;
import java.awt.event.KeyEvent;
import javax.swing.JFrame;
import javax.swing.JTextField;

public class Main {

    public static void main(String[] argv) throws Exception {

    JTextField textField = new JTextField();

    textField.addKeyListener(new Keychecker());

    JFrame jframe = new JFrame();

    jframe.add(textField);

    jframe.setSize(400, 350);

    jframe.setVisible(true);

}

class Keychecker extends KeyAdapter {

    @Override
    public void keyPressed(KeyEvent event) {

        char ch = event.getKeyChar();

        System.out.println(event.getKeyChar());

    }

}

你忘记了一个 '}'。 - FairOPShotgun

2

通用方法

我基于@Elist的方法构建了一个方便实用的工具类,可适用于任何键。

import java.awt.*;
import java.awt.event.KeyEvent;
import java.util.HashMap;
import java.util.Map;

public class Keyboard {

    private static final Map<Integer, Boolean> pressedKeys = new HashMap<>();

    static {
        KeyboardFocusManager.getCurrentKeyboardFocusManager().addKeyEventDispatcher(event -> {
            synchronized (Keyboard.class) {
                if (event.getID() == KeyEvent.KEY_PRESSED) pressedKeys.put(event.getKeyCode(), true);
                else if (event.getID() == KeyEvent.KEY_RELEASED) pressedKeys.put(event.getKeyCode(), false);
                return false;
            }
        });
    }

    public static boolean isKeyPressed(int keyCode) { // Any key code from the KeyEvent class
        return pressedKeys.getOrDefault(keyCode, false);
    }
}

使用示例:

do {
    if (Keyboard.isKeyPressed(KeyEvent.VK_W)) System.out.println("W is pressed!");
} while (!Keyboard.isKeyPressed(KeyEvent.VK_ESCAPE));

0

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接