《電子技術應用》
您所在的位置:首頁 > 嵌入式技術 > 設計應用 > 基于IOC的GUI框架設計與實現(xiàn)
基于IOC的GUI框架設計與實現(xiàn)
來源:微型機與應用2011年第10期
廖福保, 張文梅
(廣東農(nóng)工商職業(yè)技術學院 計算機科學系, 廣東 廣州510507)
摘要: 傳統(tǒng)的圖形用戶界面GUI(Graphics User Interface)設計中,存在過度耦合、組件與事件之間的映射關系混亂等問題。對此,提出了基于控制反轉(zhuǎn)(IOC)的GUI框架,該框架采用Java反射機制,解析xml配置文件完成組件實例化、組件添加事件監(jiān)聽。實驗表明,利用該框架建立的GUI實現(xiàn)了業(yè)務對象的松散耦合,組件和事件處理方法分離,縮短開發(fā)周期,具有較高的可擴展性。
Abstract:
Key words :

摘  要:傳統(tǒng)的圖形用戶界面GUI(Graphics User Interface)設計中,存在過度耦合、組件與事件之間的映射關系混亂等問題。對此,提出了基于控制反轉(zhuǎn)(IOC)的GUI框架,該框架采用Java反射機制,解析xml配置文件完成組件實例化、組件添加事件監(jiān)聽。實驗表明,利用該框架建立的GUI實現(xiàn)了業(yè)務對象的松散耦合,組件和事件處理方法分離,縮短開發(fā)周期,具有較高的可擴展性。
關鍵詞:控制反轉(zhuǎn);圖形用戶界面;Java反射機制

    Java是目前最優(yōu)秀的軟件開發(fā)語言之一,由于其結構簡單、面向?qū)ο蟆⒖缙脚_等優(yōu)越特性使它具有極強的生存力,并得到了廣泛的應用。基于Java的圖形用戶界面(GUI)中,AWT是Java提供的用來建立和設置Java圖形用戶界面的第一代開發(fā)工具。AWT由java.awt包提供,其中包含了許多可以用來建立與平臺無關的GUI類。由于AWT組件占有系統(tǒng)資源較多,常把java.awt組件稱為重量級組件。Java Swing是Java Foundation Classes(JFC)的一部分,解決了AWT的很多缺點,相對于AWT,Swing是輕量級組件。Swing提供了許多比AWT更好的屏幕顯示元素,使用純Java寫成,與Java一樣可以跨平臺運行[1]。
    圖形用戶界面(GUI)借助于多種組件,包括菜單、按鈕、文本框、選擇框、列表框等,通過相應的事件處理機制,實現(xiàn)與用戶的動態(tài)交互。
1  圖形用戶界面的建立
1.1 創(chuàng)建GUI窗口

    javax.swing.JFrame類是用來建立用戶界面的底層窗口容器,能夠容納其他組件的對象,如標簽、按鈕、文本組件等。JFrame類提供的add()方法把不同的組件添加到容器中,通過容器類的setLayout()方法可以設定容器的布局,安排各種組件在容器中。
    使用JFrame類創(chuàng)建GUI窗口的基本步驟如下:用JFrame類或其子類創(chuàng)建一個對象即窗體;設置窗口的部分屬性,如標題、寬度、高度、可見性、圖標等;添加內(nèi)容面板、組件;編寫事件處理方法;組件添加事件監(jiān)聽。
1.2 Java事件處理
    在Java中,程序與用戶的交互通過響應各種事件來實現(xiàn)。每當一個事件發(fā)生,Java虛擬機就會將事件的消息傳遞給程序,由程序中的事件處理方法對事件進行處理。Java通過委托型事件處理機制來解決對事件的響應。
    事件處理機制可表述如下[2]:事件源對象封裝了事件源、組件狀態(tài)等必要信息;當事件源對象發(fā)生改變時,向它所注冊的所有監(jiān)聽器發(fā)出通知,各監(jiān)聽器判斷事件類型是否為自己管轄范圍,若是,則通知給該監(jiān)聽器的執(zhí)行器,執(zhí)行器從事件中獲取事件信息,并執(zhí)行相應函數(shù),改變組件的狀態(tài)。
1.3 傳統(tǒng)創(chuàng)建窗口和事件處理的局限性
    在傳統(tǒng)的GUI創(chuàng)建過程中,存在一些局限性。
    (1)組件創(chuàng)建、添加都采用硬編碼方式,造成程序的過度耦合。
    (2)如果窗體中有很多組件,組件要添加注冊監(jiān)聽,則在代碼中看到很多重復注冊監(jiān)聽的代碼,而這些注冊監(jiān)聽的代碼都與界面本身設計無關,組件與事件之間的映射關系將會很混亂。
    (3)事件處理方法定義在別的類中,無法得到窗體及其組件的引用,只能得到事件源,而無法改變其他組件的狀態(tài);或者把事件處理與窗體設計放在一起,這樣程序的可維護性又不好。
    (4)不利于代碼重用,基于MVC的思想,應該把事件處理方法分離出來;在需要修改事件處理代碼時,就無需修改界面本身的源代碼。
2  圖形用戶界面設計的改進
2.1 控制反轉(zhuǎn)(IOC)

    IOC就是控制反轉(zhuǎn)[3](Inversion of Control)的縮寫,也稱為依賴注入,控制反轉(zhuǎn)IOC是一種用于控制業(yè)務對象之間依賴關系的機制,將其設計的類與類之間的關系都交由外部容器進行管理,僅需調(diào)用類在容器中注冊的名字就可以得到類的實例,有效降低了業(yè)務對象之間的依賴程度,實現(xiàn)了業(yè)務對象之間的松散耦合。
    IOC的實際意義就是把組件之間的依賴關系(調(diào)用關系)反轉(zhuǎn)出來,對象之前的依賴關系用xml配置文件描述;這樣,各個組件之間就不存在硬編碼的關聯(lián),任何組件都可以最大程度地得到重用。
    考慮如下接口和類的定義:
    public interface ICar{void operate();}
    public class Toyota implements ICar{…}
    public class Honda implements ICar{…}
    public class Driver{
        private ICar car;
        public void setCar(ICar car){this.car = car;}
        public ICar getCar(){return car;}
        public void drive(){car.operator();}
    }
    類Driver依賴于ICar,而類Toyota和Honda實現(xiàn)了接口ICar,即類Driver可以依賴于Toyota或Honda。
    運用了IOC模式后就不再需要自己管理組件之間的依賴關系,只需要聲明由xml配置文件描述去實現(xiàn)這種依賴關系,就好像把對組件之間的依賴關系的控制進行了倒置,不再由組件自己來建立這種依賴關系而是交給xml配置文件去管理。
2.2 設計的改進
       在改進的GUI編程中,把窗體中組件的創(chuàng)建、組件的外觀設置和組件觸發(fā)事件時執(zhí)行什么方法,不是以硬編碼的方式組合在一起,而是通過配置文件來配置。這樣開發(fā)人員無須關心組件的創(chuàng)建、組件的樣式設置、事件的監(jiān)聽與實現(xiàn),只需要設置相應的get、set方法來存取組件、屬性等,事件處理方法能在任意類中實現(xiàn),方法名可以自定義,并且在其他類中能夠得到窗體對象及其組件的引用。當組件的樣式發(fā)生改變時,只需改動配置文件即可。
     該改進設計通過配置文件,并利用控制反轉(zhuǎn)和Java反射機制得以實現(xiàn),這就需要有框架和良好的設計。
3 框架運行機理
     框架中各組成部分在運行過程中的調(diào)用關系如圖1所示。


    當程序入口啟動時,框架解析bean-config.xml文件;組件工廠類根據(jù)xml配置文件創(chuàng)建各種組件對象;組件外觀設置類查找xml文件為每個組件設置相應的外觀;事件監(jiān)聽器類查找xml文件為每個組件添加對應的事件監(jiān)聽器;事件執(zhí)行類查找xml文件為每個組件設置事件觸發(fā)時執(zhí)行的方法;最后還需要一個保存窗體對象的類。
        GUI程序開發(fā)人員只需要設置相應的get、set方法來存取組件,事件發(fā)生時要執(zhí)行的方法和配置xml文件。組件的建立、外觀的設置、事件監(jiān)聽添加、事件處理方法都由框架來完成。一個編碼的例子如下:
         public class JFrameDemo extends JFrame{
             private JTextField input ;
             private JButton ok ;
             //省略的get, set方法
             //省略構造方法,該方法用于添加組件到窗體
        }
         //事件處理類和方法
         public class EventOperator{
        public void operate(){
            //從保存窗體對象的類中獲得窗體
            //通過窗體的get方法獲得組件
            //執(zhí)行所需的操作并修改組件狀態(tài)
        }
     }

 


4 框架的具體實現(xiàn)
4.1 xml配置文件格式

      xml是一種標記語言,用于各種配置文件和不同語言間交換信息,它只負責信息的存儲,而不負責信息的表達。本框架bean-config.xml文件的設計格式如下:
     <?xml version="1.0" encoding="GB2312"?>
     <beans>
        <bean id="input" class="java.awt.JTextField">
            <setColumns>10;Integer</setColumns>
        </bean>
        <bean id="ok" class="java.awt.JButton">
            <setText>計算;String</setText>
            <event type="ActionListener" class="test.Event-
                Operator" method="operate"></event>
        </bean>
        <bean id="frame" class="test.JFrameDemo">
            <ref>input</ref>
            <ref>ok</ref>    
        </bean>
         </beans>
    配置文件說明如下:
    (1)根節(jié)點為beans。
    (2)bean節(jié)點中的id屬性用來唯一地標識一個組件,該值要與代碼里的組件名一致,class屬性用來表示所對應的類名。
    (3)event節(jié)點的type屬性表示監(jiān)聽器的類型, class屬性表示事件觸發(fā)時將要執(zhí)行的方法所對應的類名,method屬性表示事件觸發(fā)時將要執(zhí)行的方法。如上面xml文件中,表示當ok組件發(fā)生單擊事件時,將執(zhí)行test. EventOperator類的operate方法。
    (4)ref子節(jié)點值表示該組件需要依賴的其他bean的標識。
    (5)bean其他子節(jié)點為設置組件外觀的方法,子節(jié)點值為調(diào)用該方法所需的參數(shù)值和對應的參數(shù)類型。
4.2 Java的反射機制
    因為所對應的類、方法都保存在xml文件中,而對xml解析得到的類名和方法名都是字符串類型,要把字符串實例化成相應的對象并調(diào)用就要用到Java的反射技術[4]。
    Java的反射機制允許程序在運行時透過Reflection APIs取得任何一個已知名稱的類的內(nèi)部信息,包括其訪問權限、父類、實現(xiàn)接口,也包括成員變量和方法的所有信息,并可在運行時改變成員變量的內(nèi)容或執(zhí)行方法。
    本框架主要利用反射機制來實例化對象和調(diào)用方法。其關鍵代碼如下(className,methodName均為字符串):
       Class instance = Class.forName(className).newInstance();
                //獲得目標類實例,傳入目標類名及包名
         Class c = Class.forName(className);
         Method m = c.getMethod(methodName,new Class[]{...});
                        //傳入方法名和參數(shù)類型數(shù)組
         m.invoke(instance, new Object[]{});
        //方法執(zhí)行,傳入目標類的實例和方法參數(shù)值數(shù)組
4.3 xml文件處理器
    xml文件處理器主要用于對bean-config.xml文件進行解析, 本框架采用jdk1.5自帶的 org.w3c.dom包來解析xml文檔,為文檔對象模型(DOM) 提供接口。
     xml文件處理器根據(jù)傳入的xml文件生成Document節(jié)點,Document可看做是xml在內(nèi)存中的一個鏡像,對Document操作能夠直接同步到該xml文件。關鍵代碼如下:
    DocumentBuilderFactory dbf=DocumentBuilderFactory.new
        Instance();    
  DocumentBuilder db=dbf.newDocumentBuilder();    
                    //通過工廠得到一個DocumentBuilder    
  Document doc=db.parse("bean-config.xml");         
    //DocumentBuilder通過解析xml文件得到一個Document
4.4 組件工廠類的實現(xiàn)
    根據(jù)xml文件的bean節(jié)點建立組件對象,首先利用Document的getElementsByTagName方法獲得所有bean節(jié)點的NodeList對象,遍歷NodeList對象獲得每個bean節(jié)點的Node對象,再利用Node的getAttributes方法獲得該節(jié)點的所有屬性,然后根據(jù)獲得的id、class屬性就可以實例化組件。關鍵代碼如下:
  NodeList nodes = doc.getElementsByTagName("bean");
                                //獲得所有的bean節(jié)點
    ... ...
  Node node = nodes.item(i);//獲得其中一個bean節(jié)點        NamedNodeMap attributes = node.getAttributes();
                                     //取出該節(jié)點的所有屬性值
      ... ...
    Class cl = Class.forName(class屬性值);                     Object instance = cl.newInstance();    //創(chuàng)建該類的實例
4.5 組件外觀設置類實現(xiàn)
    從組件工廠類中獲得組件對象并從xml文件中獲得的方法名、參數(shù)值和參數(shù)類型,利用Java反射技術就可以為組件執(zhí)行方法設置組件外觀。
4.6 事件執(zhí)行類
    事件執(zhí)行類繼承多個事件接口,同時實現(xiàn)接口對應的方法。在每個實現(xiàn)的方法中,獲得xml文件中event節(jié)點的class屬性值以及method屬性值,利用Java反射技術就可以執(zhí)行方法。這時當組件觸發(fā)事件時,執(zhí)行事件執(zhí)行類的對應方法,而事件執(zhí)行類的方法是調(diào)用method屬性值的方法。這樣就實現(xiàn)了當組件觸發(fā)事件時,執(zhí)行method屬性值的方法。
     通過事件執(zhí)行類,可以自定義觸發(fā)事件時執(zhí)行的方法名,實現(xiàn)了事件監(jiān)聽與事件處理的分離。事件執(zhí)行類采用單例模式實現(xiàn)即僅有一個實例運行,節(jié)省了內(nèi)存消耗。
4.7 事件監(jiān)聽器添加類
     傳統(tǒng)GUI編程中,事件監(jiān)聽器的添加是利用組件調(diào)用相應的方法,并傳入對應的事件監(jiān)聽器對象。在本框架事件監(jiān)聽器添加類中,首先獲得event節(jié)點的type屬性值,通過Java反射技術把事件執(zhí)行類實例添加到組件中,這樣當組件觸發(fā)事件時就可以執(zhí)行事件執(zhí)行類的相關方法。
    在GUI設計中將組件設計和事件處理交予本文框架管理,降低了對象之間的依賴程度。在代碼中僅需要編寫get、set方法,也不需注冊監(jiān)聽器、實現(xiàn)接口等代碼,減少了代碼編寫量,實現(xiàn)了業(yè)務對象的松散耦合。事件觸發(fā)和事件執(zhí)行實現(xiàn)了分離,提高了程序的可維護性。對組件狀態(tài)或事件信息的改變不需修改源代碼,只需要修改配置文件,易于實現(xiàn)重構。
    實踐表明,該框架簡單易用,建立的圖形用戶界面(GUI)具有較高的靈活性、可維護性和可擴展性,對構建中小型的GUI應用具有良好的支撐作用和借鑒意義。
參考文獻
[1] 錢銀中.Java程序設計案例教程[M]. 北京:機械工業(yè)出版社,2010.
[2] 宋淼,袁兆山,陳剛.Java事件處理機制中設計模式的分析[J].安徽工業(yè)大學學報,2004,27(11):1383-1386.
[3] 魏學松,張育平.IOC框架的研究與設計[J]. 計算機技術與發(fā)展,2006,16(3):213-216.
[4] 吳其慶. Java編程思想與實踐[M]. 北京:冶金工業(yè)出版社,2006.

此內(nèi)容為AET網(wǎng)站原創(chuàng),未經(jīng)授權禁止轉(zhuǎn)載。