Posted on 2009-05-27 04:02
H2O 閱讀(568)
評論(1) 編輯 收藏 所屬分類:
java
學(xué)習(xí)過Java Swing的程序員一定對于Swing中相對較為復(fù)雜的事件驅(qū)動(dòng)模型比較困惑,雖然事件驅(qū)動(dòng)模型在Java Swing中被完完全全的體現(xiàn)出來了,但是對于一個(gè)軟件初學(xué)者而言這樣的近乎“裸體”的事件驅(qū)動(dòng)模型確實(shí)是很難理解的。
微軟公司.Net框架與Java Swing的GUI編程相比要簡單很多,同樣是事件驅(qū)動(dòng)模型.Net框架就進(jìn)行了大量的封裝處理,.Net把這種封裝稱之為委托器(Delegate),其代碼如下:
//當(dāng)btnSubmit按鈕被點(diǎn)擊以后要求交給btnSubmit_Click方法處理// EventHandler在中間啟到委托器的作用,//它負(fù)責(zé)將事件分發(fā)到指定的方法中進(jìn)行處理this.btnSubmit.Click += new EventHandler(this.btnSubmit_Click);//事件處理方法// object sender:事件源,這里指btnSubmit對象// EventArgs e:事件處理參數(shù),它保存了需要提供給程序員的必要信息private void btnSubmit_Click(object sender, EventArgs e){ //打印This is a button語句 System.Diagnostics.Debug.WriteLine("This is button");}
作為對比,我們來看看Java Swing的事件處理和委托就要復(fù)雜很多代碼如下:
//為btnSubmit增加偵聽器SelectHandler,當(dāng)btnSubmit被點(diǎn)擊以后
//有偵聽器的actionPerformed負(fù)責(zé)處理該點(diǎn)擊事件的業(yè)務(wù)
//由于事件源btnSubmit和偵聽器類SelectHandler處于兩個(gè)不同的類中
//為了讓SelectHandler類取得頁面的信息,我們需要將窗體對象(this)
//傳入到偵聽器中
btnSubmit.addActionListener(new SelectHandler(this));
//偵聽器SelectHandler,它必須實(shí)現(xiàn)動(dòng)作事件ActionListener接口
//以達(dá)到事件分發(fā)的作用
class SelectHandler implements ActionListener {
private CommonDialogDemo form = null;
//將窗體對象CommonDialogDemo通過構(gòu)造函數(shù)傳入SelectHandler類中
public SelectHandler(CommonDialogDemo form) {
this.form = form;
}
//事件處理方法,當(dāng)btnSubmit被點(diǎn)擊,自動(dòng)執(zhí)行以下打印代碼
publicvoid actionPerformed(ActionEvent e) {
System.out.println("This is button");
}
}
|
根據(jù)以上代碼,我們可以清晰的看到Java Swing要比.Net的麻煩的多,而且更不能讓人忍受的就是,一個(gè)頁面如果有多個(gè)按鈕的話,我們必須針對每個(gè)按鈕編寫多個(gè)事件偵聽類,而且這些類一般都會被設(shè)為內(nèi)部類。學(xué)過軟件建模的讀者可能知道,內(nèi)部軟件建模在軟件工程中是不推薦使用的,所以這樣的代碼編寫明顯會增加設(shè)計(jì)冗余度和復(fù)雜度,因此,我們可以考慮自己編寫一個(gè)類似于.Net中EventHandler一樣的事件委托類來處理事件分發(fā)。
由于我們無權(quán)修改Java的編譯器,所以我在這里將會借助于反射技術(shù),利用一個(gè)事件委托類處理所有的點(diǎn)擊事件,代碼如下:
package cn.softworks.teachersearchsystem.support;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.lang.reflect.Method;
/**
*該類是用來處理所有的Swing按鈕點(diǎn)擊事件,并根據(jù)將處理權(quán)<br>
*轉(zhuǎn)交給使用者來處理
*
*@authorChen.yu
*
*/
publicclass EventHandlerimplements ActionListener {
//組件所在的窗體對象
private Object form = null;
//受到委托的方法名
private String methodName = null;
/**
*構(gòu)造函數(shù)
*
*@paramform 組件所在的窗體對象
*@parammethodName 受到委托的方法名
*/
public EventHandler(Object form,String methodName) {
this.form = form;
this.methodName = methodName;
}
/**
*事件處理委托方法
*/
publicvoid actionPerformed(ActionEvent e) {
//得到窗體對象的類型
Class formType = this.form.getClass();
try {
//得到指定委托方法的類型
Method method =
formType.getMethod(this.methodName, new Class[] {e.getClass()});
//調(diào)用指定的方法
method.invoke(this.form, new Object[] {e});
}catch(Exception ex) {
return;
}
}
}
|
現(xiàn)在我們來編寫一個(gè)測試程序,代碼如下:
btnSearch.addActionListener(new EventHandler(this,"btnSearch_Click"));
public void btnSearch_Click(ActionEvent e) {
System.out.println("This is btnSearch");
}
|
從以上代碼中我們可以清晰的看到,事件處理和事件委托處于同一窗體中了,.Net方便的Delegate處理被我們用反射實(shí)現(xiàn)了。