学习过Java Swing的读者一定对于Swing中相对较为复杂的事件驱动模型比较困惑,虽然事件驱动模型在Java Swing中被完完全全的体现出来了,

但是对于一个软件初学者而言这样的近乎“裸体”的事件驱动模型确实是很难理解的。
Microsoft公司.Net框架与Java Swing的GUI编程相比要简单很多,同样是事件驱动模型.Net框架就进行了大量的封装处理,.Net把这种封装称之

为委托器(Delegate)其代码如下:


1 //当btnSubmit按钮被点击以后要求交给btnSubmit_Click方法处理
 2 // EventHandler在中间启到委托器的作用,
 3 //它负责将事件分发到指定的方法中进行处理
 4 this.btnSubmit.Click += new EventHandler(this.btnSubmit_Click);
 5 //事件处理方法
 6 // object sender:事件源,这里指btnSubmit对象
 7 // EventArgs e:事件处理参数,它保存了需要提供给程序员的必要信息
 8     private void btnSubmit_Click(object sender, EventArgs e)
 9     {
10         //打印This is a button语句
11         System.Diagnostics.Debug.WriteLine("This is button");
12    }
13

 

作为对比,我们来看看Java Swing的事件处理和委托就要复杂很多:代码如下

 


1 //为btnSubmit增加侦听器SelectHandler,当btnSubmit被点击以后
 2 //有侦听器的actionPerformed负责处理该点击事件的业务
 3 //由于事件源btnSubmit和侦听器类SelectHandler处于两个不同的类中
 4 //为了让SelectHandler类取得页面的信息,我们需要将窗体对象(this)
 5 //传入到侦听器中
 6 btnSubmit.addActionListener(new SelectHandler(this));
 7 //侦听器SelectHandler,它必须实现动作事件ActionListener接口
 8 //以达到事件分发的作用
 9 class SelectHandler implements ActionListener {
10     private CommonDialogDemo form = null;
11     //将窗体对象CommonDialogDemo通过构造函数传入SelectHandler类中
12     public SelectHandler(CommonDialogDemo form) {
13         this.form = form;
14     }
15     //事件处理方法,当btnSubmit被点击,自动执行以下打印代码
16     publicvoid actionPerformed(ActionEvent e) {
17         System.out.println("This is button");
18     }
19 }
20

 

根据以上代码,我们可以清晰的看到Java Swing要比.Net的麻烦的多,而且更不能让人忍受的就是,一个页面如果有多个按钮的话,我们必须针

对每个按钮编写多个事件侦听类,而且这些类一般都会被设为内部类。学过软件建模的读者可能知道,内部在软件建模在软件工程中是不推荐使

用的,所以这样的代码编写明显会增加设计冗余度和复杂度,因此我们可以考虑自己编写一个类似于.Net中EventHandler一样的事件委托类来处

理事件分发。
       由于我们无权修改Java的编译器,所以我在这里将会借助于反射技术,利用一个事件委托类处理所有的点击事件,代码如下:


1 package cn.softworks.teachersearchsystem.support;
 2 
 3 import java.awt.event.ActionEvent;
 4 import java.awt.event.ActionListener;
 5 import java.lang.reflect.Method;
 6 
 7 /**
 8 *该类是用来处理所有的Swing按钮点击事件,并根据将处理权<br>
 9 *转交给使用者来处理
10 *
11 *@authorChen.yu
12 *
13 */
14 publicclass EventHandlerimplements ActionListener {
15     
16     //组件所在的窗体对象
17     private Object form = null;
18     
19     //受到委托的方法名
20     private String methodName = null;
21     
22     /**
23      *构造函数
24      *
25      *@paramform           组件所在的窗体对象
26      *@parammethodName     受到委托的方法名
27      */
28     public EventHandler(Object form,String methodName) {
29         this.form = form;
30         this.methodName = methodName;
31     }
32     
33     /**
34      *事件处理委托方法
35      */
36     publicvoid actionPerformed(ActionEvent e) {
37         
38         //得到窗体对象的类型
39         Class formType = this.form.getClass();
40         
41         try {
42             //得到指定委托方法的类型 
43             Method method = 
44                 formType.getMethod(this.methodName, new Class[] {e.getClass()});
45             //调用指定的方法
46             method.invoke(this.form, new Object[] {e});
47             
48         }catch(Exception ex) {
49             
50             return;
51         }       
52     
53     }
54 
55 }
56 
57

现在我们来编写一个测试程序,代码如下:

1 btnSearch.addActionListener(
2 new EventHandler(this,"btnSearch_Click"));
3 
4 public void btnSearch_Click(ActionEvent e) {
5     System.out.println("This is btnSearch");
6 }
7

从以上代码中我们可以清晰的看到,事件处理和事件委托处于同一窗体中了,.Net方便的Delegate处理被我们用反射实现了。