反射在Java Swing中的應用

marginwidth="0" marginheight="0" src="http://www.zealware.com/csdnblog336280.html" frameborder="0" width="336" scrolling="no" height="280">
 
學習過Java Swing的讀者一定對於Swing中相對較爲複雜的事件驅動模型比較困惑,雖然事件驅動模型在Java Swing中被完完全全的體現出來了,但是對於一個軟件初學者而言這樣的近乎“裸體”的事件驅動模型確實是很難理解的。

Microsoft公司.Net框架與Java SwingGUI編程相比要簡單很多,同樣是事件驅動模型.Net框架就進行了大量的封裝處理,.Net把這種封裝稱之爲委託器(Delegate)其代碼如下:

//當btnSubmit按鈕被點擊以後要求交給btnSubmit_Click方法處理
// EventHandler在中間啓到委託器的作用,
//它負責將事件分發到指定的方法中進行處理
this.btnSubmit.Click += new EventHandler(this.btnSubmit_Click);
//事件處理方法
// object sender:事件源,這裏指btnSubmit對象
// EventArgs e:事件處理參數,它保存了需要提供給程序員的必要信息
    private void btnSubmit_Click(object sender, EventArgs e)
    {
        //打印This is a button語句
        System.Diagnostics.Debug.WriteLine("This is button");
}
 

作爲對比,我們來看看Java Swing的事件處理和委託就要複雜很多:代碼如下:(您若還不是很瞭解Swing事件驅動的話,可以參考我的另外一篇文章:事件驅動模型實例詳解(Java)

//爲btnSubmit增加偵聽器SelectHandler,當btnSubmit被點擊以後
//有偵聽器的actionPerformed負責處理該點擊事件的業務
//由於事件源btnSubmit和偵聽器類SelectHandler處於兩個不同的類中
//爲了讓SelectHandler類取得頁面的信息,我們需要將窗體對象(this)
//傳入到偵聽器中
btnSubmit.addActionListener(new SelectHandler(this));
//偵聽器SelectHandler,它必須實現動作事件ActionListener接口
//以達到事件分發的作用
class SelectHandler implements ActionListener {
    private CommonDialogDemo form = null;
    //將窗體對象CommonDialogDemo通過構造函數傳入SelectHandler類中
    public SelectHandler(CommonDialogDemo form) {
        this.form = form;
    }
    //事件處理方法,當btnSubmit被點擊,自動執行以下打印代碼
    publicvoid actionPerformed(ActionEvent e) {
        System.out.println("This is button");
    }
}

 

根據以上代碼,我們可以清晰的看到Java Swing要比.Net的麻煩的多,而且更不能讓人忍受的就是,一個頁面如果有多個按鈕的話,我們必須針對每個按鈕編寫多個事件偵聽類,而且這些類一般都會被設爲內部類。學過軟件建模的讀者可能知道,內部在軟件建模在軟件工程中是不推薦使用的,所以這樣的代碼編寫明顯會增加設計冗餘度和複雜度,因此我們可以考慮自己編寫一個類似於.NetEventHandler一樣的事件委託類來處理事件分發。

       由於我們無權修改Java的編譯器,所以我在這裏將會藉助於反射技術,利用一個事件委託類處理所有的點擊事件,代碼如下:

package cn.softworks.teachersearchsystem.support;
 
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.lang.reflect.Method;
 
/**
 *該類是用來處理所有的Swing按鈕點擊事件,並根據將處理權
 *轉交給使用者來處理
 *
 *@authorChen.yu
 *
 */
publicclass EventHandlerimplements ActionListener {
   
    //組件所在的窗體對象
    private Object form = null;
   
    //受到委託的方法名
    private String methodName = null;
   
    /**
     *構造函數
     *
     *@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()});
            //調用指定的方法
            method.invoke(this.form, new Object[] {e});
           
        }catch(Exception ex) {
           
            return;
        }      
   
    }
 
}
 
現在我們來編寫一個測試程序,代碼如下:
       btnSearch.addActionListener(
new EventHandler(this,"btnSearch_Click"));
 
public void btnSearch_Click(ActionEvent e) {
    System.out.println("This is btnSearch");
}
 
從以上代碼中我們可以清晰的看到,事件處理和事件委託處於同一窗體中了,.Net方便的Delegate處理被我們用反射實現了。  


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章