485.md 13.6 KB
Newer Older
W
init  
wizardforcel 已提交
1 2 3 4
# 关于编写事件监听器的一般信息

> 原文: [https://docs.oracle.com/javase/tutorial/uiswing/events/generalrules.html](https://docs.oracle.com/javase/tutorial/uiswing/events/generalrules.html)

W
wizardforcel 已提交
5
本节讨论在应用程序中实现事件处理器时要记住的几个设计注意事项。然后,我们向您介绍描述每个事件的事件对象小对象。特别是,我们讨论`EventObject`,它是所有 AWT 和 Swing 事件的超类。接下来,我们介绍低级事件和语义事件的概念,建议您在可能的情况下更喜欢语义事件。本节的其余部分讨论了您可能在某些事件监听器中使用的实现技术,或者在由其他人或 GUI 构建器创建的事件监听器中查看的实现技术。
W
init  
wizardforcel 已提交
6 7 8 9 10 11 12 13 14 15

*   [设计注意事项](#design)
*   [获取活动信息:活动对象](#eventobjects)
*   [概念:低级事件和语义事件](#twokinds)
*   [事件适配器](#eventAdapters)
*   [内部类和匿名内部类](#innerClasses)
*   [EventHandler 类](#eventHandlers)

关于事件监听器要记住的最重要的规则是它们应该非常快速地执行。因为所有绘图和事件监听方法都在同一个线程中执行,所以一个缓慢的事件监听器方法可能会使程序看起来没有响应而且重绘速度很慢。如果由于事件需要执行一些冗长的操作,请通过启动另一个线程(或以某种方式向另一个线程发送请求)来执行操作。有关使用线程的帮助,请参阅 Swing 中的[并发。](../concurrency/index.html)

W
wizardforcel 已提交
16
您有很多关于如何实现事件监听器的选择。我们不能推荐一种特定的方法,因为一种解决方案不适合所有情况。但是,即使您没有在程序中使用相同的解决方案,我们也可以为您提供一些提示并向您展示您可能会看到的一些技巧。
W
init  
wizardforcel 已提交
17

W
wizardforcel 已提交
18
例如,您可以选择为不同类型的事件监听器实现单独的类。这可以是一个易于维护的架构,但许多类也可能意味着降低性能。
W
init  
wizardforcel 已提交
19

W
wizardforcel 已提交
20
在设计程序时,您可能希望在不公开的类中实现事件监听器,但更隐藏的地方。私有实现是一种更安全的实现。
W
init  
wizardforcel 已提交
21 22 23 24 25 26 27 28 29 30 31

如果您有一种非常特殊的简单事件监听器,则可以通过使用`EventHandler`类来避免创建类。

每个事件监听器方法都有一个参数,一个继承自 [`EventObject`](https://docs.oracle.com/javase/8/docs/api/java/util/EventObject.html) 类的对象。尽管参数总是来自`EventObject`,但其类型通常更精确地指定。例如,处理鼠标事件的方法的参数是`MouseEvent`的实例,其中`MouseEvent``EventObject`的间接子类。

`EventObject`类定义了一个非常有用的方法:

**`Object getSource()`**

Returns the object that fired the event.

W
wizardforcel 已提交
32
请注意,`getSource`方法返回`Object`。事件类有时会定义类似于`getSource`的方法,但它们具有更多受限制的返回类型。例如,`ComponentEvent`类定义`getComponent`方法,就像`getSource`返回触发事件的对象一样。区别在于`getComponent`总是返回`Component`。事件监听器的每个操作方法页面都提到您是否应该使用`getSource`或其他方法来获取事件源。
W
init  
wizardforcel 已提交
33 34 35

通常,事件类定义返回有关事件的信息的方法。例如,您可以查询`MouseEvent`对象,以获取有关事件发生位置,用户点击次数,按下了哪些修改键等信息。

W
wizardforcel 已提交
36
事件可以分为两组:*低级*事件和*语义*事件。低级事件表示窗口系统发生或低级输入。其他一切都是语义事件。
W
init  
wizardforcel 已提交
37 38 39

低级事件的示例包括鼠标和键事件,这两者都直接来自用户输入。语义事件的示例包括动作和项目事件。语义事件可能由用户输入触发;例如,按钮通常在用户单击时触发动作事件,并且当用户按下 _Enter_ 时,文本字段触发动作事件。但是,一些语义事件根本不是由低级别事件触发的。例如,当表模型从数据库接收新数据时,可能会触发表模型事件。

W
wizardforcel 已提交
40
只要有可能,您应该监听语义事件而不是低级别事件。这样,您可以使代码尽可能健壮且可移植。例如,监听按钮上的动作事件而不是鼠标事件意味着当用户尝试使用键盘替代或特定于外观的手势激活按钮时,按钮将作出适当的反应。在处理复合组件(如组合框)时,必须坚持语义事件,因为您没有可靠的方法在可能用于构成复合的所有特定于外观的组件上注册监听器零件。
W
init  
wizardforcel 已提交
41

W
wizardforcel 已提交
42
某些监听器接口包含多个方法。例如,`MouseListener`接口包含五种方法:`mousePressed``mouseReleased``mouseEntered``mouseExited``mouseClicked`。即使您只关心鼠标点击,如果您的类直接实现`MouseListener`,那么您必须实现所有五种`MouseListener`方法。您不关心的事件的方法可能有空体。这是一个例子:
W
init  
wizardforcel 已提交
43

W
wizardforcel 已提交
44
```java
W
init  
wizardforcel 已提交
45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
//An example that implements a listener interface directly.
public class MyClass implements MouseListener {
    ...
        someObject.addMouseListener(this);
    ...
    /* Empty method definition. */
    public void mousePressed(MouseEvent e) {
    }

    /* Empty method definition. */
    public void mouseReleased(MouseEvent e) {
    }

    /* Empty method definition. */
    public void mouseEntered(MouseEvent e) {
    }

    /* Empty method definition. */
    public void mouseExited(MouseEvent e) {
    }

    public void mouseClicked(MouseEvent e) {
        ...//Event listener implementation goes here...
    }
}

```

W
wizardforcel 已提交
73
由此产生的空方法体集合可以使代码更难以阅读和维护。为了帮助您避免实现空方法体,API 通常包含一个*适配器*类,用于具有多个方法的每个监听器接口。 ([监听器 API 表](api.html)列出了所有监听器及其适配器。)例如,`MouseAdapter`类实现`MouseListener`接口。适配器类实现其所有接口方法的空版本。
W
init  
wizardforcel 已提交
74

W
wizardforcel 已提交
75
要使用适配器,您需要创建它的子类并仅覆盖感兴趣的方法,而不是直接实现监听器接口的所有方法。以下是修改前面代码以扩展`MouseAdapter`的示例。通过扩展`MouseAdapter`,它继承了`MouseListener`包含的所有五种方法的空定义。
W
init  
wizardforcel 已提交
76

W
wizardforcel 已提交
77
```java
W
init  
wizardforcel 已提交
78 79 80 81 82 83 84 85 86 87 88 89 90 91 92
/*
 * An example of extending an adapter class instead of
 * directly implementing a listener interface.
 */
public class MyClass extends MouseAdapter {
    ... 
        someObject.addMouseListener(this);
    ... 
    public void mouseClicked(MouseEvent e) {
        ...//Event listener implementation goes here...
    }
}

```

W
wizardforcel 已提交
93
如果要使用适配器类,但不希望公共类从适配器类继承,该怎么办?例如,假设您编写了一个 applet,并且希望`Applet`子类包含一些代码来处理鼠标事件。由于 Java 语言不允许多重继承,因此您的类不能扩展`Applet``MouseAdapter`类。一个解决方案是在`Applet`子类中定义一个*内部类*,它扩展了`MouseAdapter`类。
W
init  
wizardforcel 已提交
94

W
wizardforcel 已提交
95
内部类对于直接实现一个或多个接口的事件监听器也很有用。
W
init  
wizardforcel 已提交
96

W
wizardforcel 已提交
97
```java
W
init  
wizardforcel 已提交
98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119
//An example of using an inner class.
public class MyClass extends Applet {
    ...
        someObject.addMouseListener(new MyAdapter());
    ...
    class MyAdapter extends MouseAdapter {
        public void mouseClicked(MouseEvent e) {
            ...//Event listener implementation goes here...
        }
    }
}

```

* * *

**Performance note:** 

在考虑是否使用内部类时,请记住,应用程序启动时间和内存占用量通常与您加载的类数成正比。您创建的类越多,启动程序所需的时间越长,所需的内存就越多。作为应用程序开发人员,您必须将其与您可能具有的其他设计约束相平衡。我们并不是建议您将应用程序转换为单个整体类,以期缩短启动时间和内存占用,这将导致不必要的麻烦和维护负担。

* * *

W
wizardforcel 已提交
120
您可以创建内部类而不指定名称,这称为*匿名内部类*。虽然初看起来可能看起来很奇怪,但匿名内部类可以使代码更容易阅读,因为类被定义在引用它的位置。但是,您需要权衡方便性与增加类数量可能带来的性能影响。
W
init  
wizardforcel 已提交
121 122 123

以下是使用匿名内部类的示例:

W
wizardforcel 已提交
124
```java
W
init  
wizardforcel 已提交
125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146
//An example of using an anonymous inner class.
public class MyClass extends Applet {
    ...
        someObject.addMouseListener(new MouseAdapter() {
            public void mouseClicked(MouseEvent e) {
                ...//Event listener implementation goes here...
            }
        });
    ...
    }
}

```

* * *

**Note:** 

匿名内部类的一个缺点是长期持久性机制无法看到它们。有关更多信息,请参阅 [JavaBeans](../../javabeans/) 中的 [JavaBeans™软件包](https://docs.oracle.com/javase/8/docs/api/java/beans/package-summary.html#package_description)[Bean 持久性](../../javabeans/advanced/persistence.html)课程的 API 文档。

* * *

W
wizardforcel 已提交
147
即使您的事件监听器需要访问封闭类中的私有实例变量,内部类也可以工作。只要您不将内部类声明为`static`,内部类就可以引用实例变量和方法,就好像它的代码在包含类中一样。要使局部变量可用于内部类,只需将变量的副本保存为`final`局部变量。
W
init  
wizardforcel 已提交
148

W
wizardforcel 已提交
149
要引用封闭实例,可以使用`_EnclosingClass_ 。这`。有关内部类的更多信息,请参见[嵌套类](../../java/javaOO/nested.html)
W
init  
wizardforcel 已提交
150

W
wizardforcel 已提交
151
[`EventHandler`](https://docs.oracle.com/javase/8/docs/api/java/beans/EventHandler.html) 类支持动态生成简单的单语句事件监听器。虽然`EventHandler`仅对特定类型的极其简单的事件监听器有用,但值得一提的是有两个原因。它对以下内容很有用:
W
init  
wizardforcel 已提交
152 153 154 155 156 157 158 159 160 161

*   创建一个持久化可以看到的事件监听器,但不会使用事件监听器接口和方法阻塞自己的类。
*   不添加应用程序中定义的类数可以帮助提高性能。

手动创建`EventHandler`很困难。必须仔细构建`EventHandler`。如果你犯了一个错误,你不会在编译时得到通知,它会在运行时抛出一个模糊的异常。因此,`EventHandler`最好由 GUI 构建器创建。 `EventHandler`应该仔细记录。否则,您将面临生成难以阅读的代码的风险。

`EventHandler`类旨在供交互式工具(如应用程序构建器)使用,这些工​​具允许开发人员在 bean 之间建立连接。通常,连接是从用户界面 bean(事件源)到应用程序逻辑 bean(目标)。这种最有效的连接将应用程序逻辑与用户界面隔离开来。例如,从 JCheckBox 到接受布尔值的方法的连接的`EventHandler`可以处理提取复选框的状态并将其直接传递给方法,以便该方法与用户界面层隔离。

内部类是处理来自用户界面的事件的另一种更通用的方法。 `EventHandler`类仅处理使用内部类可能的子集。但是,`EventHandler`在长期持久性方案中比内部类更好。此外,在大型应用程序中使用`EventHandler`,其中多次实现相同的接口可以减少应用程序的磁盘和内存占用。

W
wizardforcel 已提交
162
使用`EventHandler`的示例`EventHandler`的最简单用法是安装一个监听器,该监听器在没有参数的情况下调用目标对象上的方法。在下面的示例中,我们创建一个 ActionListener,它在`javax.swing.JFrame`的实例上调用 toFront 方法。
W
init  
wizardforcel 已提交
163

W
wizardforcel 已提交
164
```java
W
init  
wizardforcel 已提交
165 166 167 168 169 170 171
    myButton.addActionListener(
        (ActionListener)EventHandler.create(ActionListener.class, frame, "toFront"));

```

按下 myButton 时,将执行语句 frame.toFront()。通过定义 ActionListener 接口的新实现并将其实例添加到按钮,可以获得相同的效果,并具有一些额外的编译时类型安全性:

W
wizardforcel 已提交
172
```java
W
init  
wizardforcel 已提交
173 174 175 176 177 178 179 180 181
    //Equivalent code using an inner class instead of EventHandler.
    myButton.addActionListener(new ActionListener() {
        public void actionPerformed(ActionEvent e) {
            frame.toFront();
        }
    });

```

W
wizardforcel 已提交
182
`EventHandler`的下一个最简单的用法是从监听器接口中的方法的第一个参数(通常是事件对象)中提取属性值,并使用它来设置目标对象中属性的值。在下面的示例中,我们创建一个 ActionListener,它将 target(myButton)对象的 nextFocusableComponent 属性设置为事件的“source”属性的值。
W
init  
wizardforcel 已提交
183

W
wizardforcel 已提交
184
```java
W
init  
wizardforcel 已提交
185 186 187 188 189 190
    EventHandler.create(ActionListener.class, myButton, "nextFocusableComponent", "source")

```

这将对应于以下内部类实现:

W
wizardforcel 已提交
191
```java
W
init  
wizardforcel 已提交
192 193 194 195 196 197 198 199 200 201 202
    //Equivalent code using an inner class instead of EventHandler.
    new ActionListener() {
        public void actionPerformed(ActionEvent e) {
            myButton.setNextFocusableComponent((Component)e.getSource()); 
        }
    }

```

也可以创建一个只将传入事件对象传递给目标操作的`EventHandler`。如果第四个`EventHandler.create`参数是一个空字符串,那么事件只是传递:

W
wizardforcel 已提交
203
```java
W
init  
wizardforcel 已提交
204 205 206 207 208 209
    EventHandler.create(ActionListener.class, target, "doActionEvent", "")

```

这将对应于以下内部类实现:

W
wizardforcel 已提交
210
```java
W
init  
wizardforcel 已提交
211 212 213 214 215 216 217 218 219 220 221
    //Equivalent code using an inner class instead of EventHandler.
    new ActionListener() {
        public void actionPerformed(ActionEvent e) {
            target.doActionEvent(e);
        }
    }

```

可能`EventHandler`最常见的用法是从事件对象的源中提取属性值,并将此值设置为目标对象的属性值。在下面的示例中,我们创建一个 ActionListener,它将目标对象的“label”属性设置为事件源的“text”属性值(“source”属性的值)。

W
wizardforcel 已提交
222
```java
W
init  
wizardforcel 已提交
223 224 225 226 227 228
    EventHandler.create(ActionListener.class, myButton, "label", "source.text")

```

这将对应于以下内部类实现:

W
wizardforcel 已提交
229
```java
W
init  
wizardforcel 已提交
230 231 232 233 234 235 236 237
    //Equivalent code using an inner class instead of EventHandler.
    new ActionListener {
        public void actionPerformed(ActionEvent e) {
            myButton.setLabel(((JTextField)e.getSource()).getText()); 
        }
    }

```