Java与C#的事件处理都是实现了事件源-事件响应者机制,但又不完全相同。Java实现的是一种事件源与事件响应者两级实体对象方式,这里的事件响应者也是事件监听者,而C#实现的是一种事件源-代理-事件响应者三级实体对象方式。下面就这两种方式来具体说明。
Java事件处理
从概念上讲,事件是一种在"源对象"和"监听者对象"之间,某种状态发生变化的传递机制。事件有许多不同的用途,例如在Windows系统中常要处理的鼠标事件、窗口边界改变事件、键盘事件等。在Java中则是定义了一个普通的、可扩充的事件机制,这种机制能够:
对事件类型和传递的模型的定义和扩充提供一个公共框架,并适合于广泛的应用。
与Java语言和环境有较高的集成度。
事件能被描述环境捕获和触发。
能使其它构造工具采取某种技术在设计时直接控制事件,以及事件源和事件监听者之间的联系。
事件机制本身不依赖于复杂的开发工具。
事件从事件源到监听者的传递是通过对目标监听者对象的Java方法调用进行的。 对每个明确的事件的发生,都相应地定义一个明确的Java方法。这些方法都集中定义在事件监听者(EventListener)接口中,这个接口要继承java.util.EventListener。实现了事件监听者接口中一些或全部方法的类就是事件监听者。 伴随着事件的发生,相应的状态通常都封装在事件状态对象中,该对象必须继承自java.util.EventObject。事件状态对象作为单参传递给应响应该事件的监听者方法中。发出某种特定事件的事件源的标识是:遵从规定的设计格式为事件监听者定义注册方法,并接受对指定事件监听者接口实例的引用。有时,事件监听者不能直接实现事件监听者接口,或者还有其它的额外动作时,就要在一个源与其它一个或多个监听者之间插入一个事件适配器类的实例,来建立它们之间的联系。
事件状态对象(Event State Object)
与事件发生有关的状态信息一般都封装在一个事件状态对象中,这种对象是java。util。EventObject的子类。按设计习惯,这种事件状态对象类的名应以Event结尾。例如:
public class MouseMovedExampleEvent extends java。util。EventObject { protected int x, y; /* 创建一个鼠标移动事件MouseMovedExampleEvent */ MouseMovedExampleEvent(java.awt.Component source, Point location) { super(source); x = location.x; y = location.y; } /* 获取鼠标位置*/ public Point getLocation() { return new Point(x, y); }} |
/*先定义了一个鼠标移动事件对象*/ public class MouseMovedExampleEvent extends java。util。EventObject { // 在此类中包含了与鼠标移动事件有关的状态信息 ... } /*定义了鼠标移动事件的监听者接口*/ interface MouseMovedExampleListener extends java。util。EventListener { /*在这个接口中定义了鼠标移动事件监听者所应支持的方法*/ void mouseMoved(MouseMovedExampleEvent mme); } |
class ArbitraryObject implements MouseMovedExampleListener { public void mouseMoved(MouseMovedExampleEvent mme) { ... } } |
public void add< ListenerType>(< ListenerType> listener); public void remove< ListenerType>(< ListenerType> listener); |
public interface ModelChangedListener extends java。util。EventListener { void modelChanged(EventObject e); } |
public abstract class Model { private Vector listeners = new Vector(); // 定义了一个储存事件监听者的数组 /*上面设计格式中的< ListenerType>在此处即是下面的ModelChangedListener*/ public synchronized void addModelChangedListener(ModelChangedListener mcl) { listeners.addElement(mcl); }//把监听者注册入listeners数组中 public synchronized void removeModelChangedListener(ModelChangedListener mcl) { listeners.removeElement(mcl); file://把监听者从listeners中注销 } /*以上两个方法的前面均冠以synchronized,是因为运行在多线程环境时,可能同时有几个对象同时要进行注册和注销操作,使用synchronized来确保它们之间的同步。开发工具或程序员使用这两个方法建立源与监听者之间的事件流*/ protected void notifyModelChanged() {/**事件源使用本方法通知监听者发生了modelChanged事件*/ Vector l; EventObject e = new EventObject(this); /* 首先要把监听者拷贝到l数组中,冻结EventListeners的状态以传递事件。这样来确保在事件传递到所有监听者之前,已接收了事件的目标监听者的对应方法暂不生效。*/ synchronized(this) { l = (Vector)listeners.clone(); } for (int i = 0; i < l.size(); i++) { /* 依次通知注册在监听者队列中的每个监听者发生了modelChanged事件, 并把事件状态对象e作为参数传递给监听者队列中的每个监听者*/ ((ModelChangedListener)l.elementAt(i)).modelChanged(e); } } } |
temp。Click+=new System.EventHandler(this.Test);//为test添加事件处理方法 |
public delegate void EventHandler(object sender,EventArgs e); |
private void button1_Click(object sender, System.EventArgs e) |
private void button1_Click(object sender, System。EventArgs e) |
delegate int MyEventHandler(object sender, ToolBarButtonClickEventArgs e); |
private int MyTest(object sender,ToolBarButtonClickEventArgs e) {} |
Control。Event+=new MyEventHandler(MyTest); |