学习过Java Swing的读者一定对于Swing中相对较为复杂的事件驱动模型比较困惑虽然事件驱动模型在Java Swing中被完完全全的体现出来了但是对于一个软件初学者而言这样的近乎裸体的事件驱动模型确实是很难理解的
Microsoft公司Net框架与Java Swing的GUI编程相比要简单很多同样是事件驱动模型Net框架就进行了大量的封装处理Net把这种封装称之为委托器(Delegate)其代码如下
//当btnSubmit按钮被点击以后要求交给btnSubmit_Click方法处理
// EventHandler在中间启到委托器的作用
//它负责将事件分发到指定的方法中进行处理
thisbtnSubmitClick += new EventHandler(thisbtnSubmit_Click);
//事件处理方法
// object sender:事件源这里指btnSubmit对象
// EventArgs e:事件处理参数它保存了需要提供给程序员的必要信息
private void btnSubmit_Click(object sender EventArgs e)
{
//打印This is a button语句
SystemDiagnosticsDebugWriteLine(This is button);
}
作为对比我们来看看Java Swing的事件处理和委托就要复杂很多代码如下(您若还不是很了解Swing事件驱动的话可以参考我的另外一篇文章事件驱动模型实例详解(Java篇))
//为btnSubmit增加侦听器SelectHandler当btnSubmit被点击以后
//有侦听器的actionPerformed负责处理该点击事件的业务
//由于事件源btnSubmit和侦听器类SelectHandler处于两个不同的类中
//为了让SelectHandler类取得页面的信息我们需要将窗体对象(this)
//传入到侦听器中
btnSubmitaddActionListener(new SelectHandler(this));
//侦听器SelectHandler它必须实现动作事件ActionListener接口
//以达到事件分发的作用
class SelectHandler implements ActionListener {
private CommonDialogDemo form = null;
//将窗体对象CommonDialogDemo通过构造函数传入SelectHandler类中
public SelectHandler(CommonDialogDemo form) {
thisform = form;
}
//事件处理方法当btnSubmit被点击自动执行以下打印代码
publicvoid actionPerformed(ActionEvent e) {
Systemoutprintln(This is button);
}
}
根据以上代码我们可以清晰的看到Java Swing要比Net的麻烦的多而且更不能让人忍受的就是一个页面如果有多个按钮的话我们必须针对每个按钮编写多个事件侦听类而且这些类一般都会被设为内部类学过软件建模的读者可能知道内部在软件建模在软件工程中是不推荐使用的所以这样的代码编写明显会增加设计冗余度和复杂度因此我们可以考虑自己编写一个类似于Net中EventHandler一样的事件委托类来处理事件分发
由于我们无权修改Java的编译器所以我在这里将会借助于反射技术利用一个事件委托类处理所有的点击事件代码如下
package cnsoftworksteachersearchsystemsupport;
import javaawteventActionEvent;
import javaawteventActionListener;
import javalangreflectMethod;
/**
*该类是用来处理所有的Swing按钮点击事件并根据将处理权<br>
*转交给使用者来处理
*
*@authorChenyu
*
*/
publicclass EventHandlerimplements ActionListener {
//组件所在的窗体对象
private Object form = null;
//受到委托的方法名
private String methodName = null;
/**
*构造函数
*
*@paramform 组件所在的窗体对象
*@parammethodName 受到委托的方法名
*/
public EventHandler(Object formString methodName) {
thisform = form;
thodName = methodName;
}
/**
*事件处理委托方法
*/
publicvoid actionPerformed(ActionEvent e) {
//得到窗体对象的类型
Class formType = thisformgetClass();
try {
//得到指定委托方法的类型
Method method =
formTypegetMethod(thodName new Class[] {egetClass()});
//调用指定的方法
methodinvoke(thisform new Object[] {e});
}catch(Exception ex) {
return;
}
}
}
现在我们来编写一个测试程序代码如下
btnSearchaddActionListener(new EventHandler(thisbtnSearch_Click));
public void btnSearch_Click(ActionEvent e) {
Systemoutprintln(This is btnSearch);
}
从以上代码中我们可以清晰的看到事件处理和事件委托处于同一窗体中了Net方便的Delegate处理被我们用反射实现了