KeyListener无法解决的问题

时间:2016-04-28 20:54:03

标签: java keylistener runnable

虽然我已经编写了很长一段时间,但我似乎无法解决我遇到的这个问题(我几乎肯定它是一个我无法捕获的小语法错误而不是逻辑错误)。我正在制作一个音板,并且正在使用键盘上的数字键盘。我有一个KeyManager类:

public class KeyManager implements KeyListener{


private boolean[] keys;
private boolean nine, eight, seven, six, five, four, three, two, one; 

public KeyManager(){
    keys = new boolean [256];
}

public void update(){
    nine = keys[KeyEvent.VK_9];
    eight = keys[KeyEvent.VK_8];
    seven = keys[KeyEvent.VK_7];
    six = keys[KeyEvent.VK_6];
    five = keys[KeyEvent.VK_5];
    four = keys[KeyEvent.VK_4];
    three = keys[KeyEvent.VK_3];
    two = keys[KeyEvent.VK_2];
    one = keys[KeyEvent.VK_1];
}



public void keyPressed(KeyEvent e) {
    keys[e.getKeyCode()] = true;
}

public void keyReleased(KeyEvent e) {
    keys[e.getKeyCode()] = false;
}

public void keyTyped(KeyEvent e) {

}
public boolean isNine() {
    return nine;
}

public boolean isEight() {
    return eight;
}

public boolean isSeven() {
    return seven;
}

public boolean isSix() {
    return six;
}

public boolean isFive() {
    return five;
}

public boolean isFour() {
    return four;
}

public boolean isThree() {
    return three;
}

public boolean isTwo() {
    return two;
}

public boolean isOne() {
    return one;
}
}

这很简单,我正在使用一个将JFrame扩展为主要UI显示的Frame类:

public class Frame extends JFrame implements Runnable{
private static final long serialVersionUID = 1L;
private Pane pane;
private ArrayList<Circle> c;
private KeyManager keyManager;

public Frame(){
    keyManager = new KeyManager();
    this.addKeyListener(keyManager);
    this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    this.getContentPane().setPreferredSize(new Dimension(400, 400));
    this.pack();
    this.setLocationRelativeTo(null);
    pane = new Pane();
    this.add(pane);
    c = new ArrayList<Circle>();
    c.add(new Circle(20, 140));
    c.add(new Circle(80, 140));
    c.add(new Circle(140, 140));
    c.add(new Circle(20, 80));
    c.add(new Circle(80, 80));
    c.add(new Circle(140, 80));
    c.add(new Circle(20, 20));
    c.add(new Circle(80, 20));
    c.add(new Circle(140, 20));
    this.setVisible(true);
}

public void init(){
    pane.draw(c);
}

public void update(){
    pane.draw(c);
    keyManager.update();
    if(keyManager.isNine()){
        System.out.println("pressed");
        c.get(8).press();
    }
    else{
        c.get(8).release();
        System.out.println("unpressed");
    }
}


public void run() {

    init();

    int fps = 60;
    double timePerUpdate = 1000000000/fps;
    double delta = 0;
    long now;
    long lastTime= System.nanoTime();
    long timer = 0;
    int updates = 0;

    while(true){
        now = System.nanoTime();
        delta += (now - lastTime) / timePerUpdate;
        timer+= now - lastTime; 
        lastTime = now;

        if(delta >= 1){
            update();
            updates ++;
            delta--;
        }

        if(timer >= 1000000000){
            //System.out.println("Updates and Frames: "+ updates);
            updates = 0;
            timer = 0;
        }

    }
}


}

出于某种原因,当我按下键盘上的“九”时,它不会在Frame.update()方法中按下,而只会打印出“未按下”。我迷失了导致这个问题的原因。这是我第一次使用KeyManager类而不是让Frame完成所有提升,因为它是一个很好的编程实践,但从逻辑上说它应该可以工作,除非我遗漏了一些东西......

1 个答案:

答案 0 :(得分:1)

看起来你根本没有注册你的密钥管理器。 JFrame如何知道它有一个关键的监听器?

在Frame类this.addKeyListener(new KeyManager());

的最顶部添加此行代码

像这样:

public Frame(){
    this.addKeyListener(new KeyManager()); // New code
    this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    this.getContentPane().setPreferredSize(new Dimension(400, 400));

编辑:为了将来参考,请务必检查javadoc,因为在这种情况下答案就在那里:

  

然后使用a注册从该类创建的侦听器对象   组件使用组件的 addKeyListener 方法。

的Javadoc: https://docs.oracle.com/javase/7/docs/api/java/awt/event/KeyListener.html

官方教程中再次提及addKeyListener

https://docs.oracle.com/javase/tutorial/uiswing/events/keylistener.html

编辑评论:

让我们通过在按键事件中添加一些代码来调试问题。通过这样做,我们可以找到几件事。

1)关键事件是否被解雇。

2)按下的关键代码和字符是什么

public void keyPressed(KeyEvent e) {
    System.out.println("The following key was pressed: "+e.getKeyChar());
    System.out.println("With the following key code: "+e.getKeyCode());
    System.out.println("Is that the expected output?");
    keys[e.getKeyCode()] = true;
}

您是否从我们的新调试代码中获得了正确的输出?键码是否正确,字符是否正确?你的其他数字键有用吗?

这里要记住的一件事是,数字键盘号码与普通数字键的键码不同。我强烈建议您重新编写代码以使用e.getKeyChar()或在update()方法中包含小键盘密钥代码,并在按下普通或数字键盘时使用IF语句并将布尔值设置为true /假的。