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

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

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

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

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


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

 


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

此內容為AET網站原創,未經授權禁止轉載。
亚洲一区二区欧美_亚洲丝袜一区_99re亚洲国产精品_日韩亚洲一区二区
在线观看视频日韩| 久久一日本道色综合久久| 国产日韩成人精品| 午夜精品99久久免费| 午夜精品久久久久久| 国产日本欧洲亚洲| 欧美三日本三级三级在线播放| 亚洲欧美在线一区二区| 一区二区三区欧美成人| 亚洲一区欧美激情| 国产色产综合产在线视频| 久久亚洲图片| 久久久久久久久久码影片| 亚洲欧洲综合另类| 日韩视频免费在线| 国产麻豆91精品| 美女精品视频一区| 亚洲一区二区三区四区五区午夜| 欧美影院精品一区| 亚洲黄色一区二区三区| 国产精品久久| 久久亚洲美女| 久久久综合精品| 久久成人资源| 一区二区三区蜜桃网| 一本到12不卡视频在线dvd| 午夜激情综合网| 亚洲欧美久久久久一区二区三区| 一区二区三区在线看| 欧美视频中文字幕| 欧美性jizz18性欧美| 久久视频在线免费观看| 久久久国产精品一区二区中文 | 欧美色图麻豆| 国产精品狠色婷| 国产日产高清欧美一区二区三区| 国产日韩欧美亚洲一区| 国产一区日韩一区| 欧美激情一区二区三区在线视频观看 | 91久久久在线| 日韩亚洲国产欧美| 亚洲欧美制服另类日韩| 午夜影视日本亚洲欧洲精品| 久久本道综合色狠狠五月| 亚洲日本一区二区三区| 国产亚洲精品美女| 曰本成人黄色| 日韩写真在线| 亚洲丰满少妇videoshd| 国产亚洲精品久| 狠狠色综合网| 国产欧美日韩一区| 国外成人性视频| 国产精品一区二区女厕厕| 国产一区美女| 亚洲啪啪91| 亚洲欧美春色| 亚洲精品一区二区网址| 久久国产精品久久久久久电车| 亚洲国产高清一区| 欧美在线播放视频| 亚洲精品一区二区三区蜜桃久| 亚洲一区二区三区四区五区黄| 久久久精品国产免大香伊 | 欧美激情一区二区三级高清视频| 国产精品成人久久久久| 国产一区二区三区无遮挡| 亚洲黄色天堂| 午夜国产精品视频| 夜夜爽99久久国产综合精品女不卡| 欧美一区二区三区四区高清| 一本色道久久综合亚洲91| 午夜激情一区| 欧美国产日韩一区二区| 久久夜色精品国产欧美乱极品| 欧美人在线视频| 欧美日韩国产精品一卡| 男人的天堂亚洲在线| 老司机aⅴ在线精品导航| 久久免费视频在线观看| 欧美日韩精品一区二区天天拍小说 | 一本色道久久综合亚洲精品不 | 亚洲黄色大片| 亚洲激情综合| 亚洲欧美在线aaa| 欧美激情视频网站| 国产在线不卡精品| 亚洲欧美成人网| 在线视频精品一| 免费在线观看成人av| 国产亚洲精品美女| 亚洲亚洲精品在线观看| 亚洲午夜未删减在线观看| 亚洲区一区二| 久久色在线观看| 国产人妖伪娘一区91| 一区电影在线观看| 日韩视频在线一区| 免费不卡亚洲欧美| 黄色亚洲网站| 欧美亚洲一级| 欧美在线看片a免费观看| 一区二区三区高清| 欧美激情影音先锋| 亚洲高清一二三区| 日韩手机在线导航| 亚洲国产精品高清久久久| 欧美一级久久| 国产精品久久77777| 99视频热这里只有精品免费| 9l视频自拍蝌蚪9l视频成人| 免费成人高清视频| 精品动漫3d一区二区三区免费| 欧美一区二区三区免费观看| 欧美一区二区三区在线看| 国产精品久久久久国产精品日日| 亚洲精品自在久久| 一本久道久久综合狠狠爱| 欧美日韩高清在线观看| 亚洲美女电影在线| 99精品视频网| 欧美日本视频在线| 99亚洲一区二区| 亚洲在线一区二区| 欧美亚一区二区| 亚洲视频图片小说| 久久精品导航| 亚洲调教视频在线观看| 久久精品国产亚洲5555| 国产日产欧美一区| 午夜国产精品视频| 久久久久一区二区三区| 国产资源精品在线观看| 久久精品论坛| 欧美mv日韩mv国产网站app| 亚洲级视频在线观看免费1级| 亚洲精品乱码久久久久久黑人| 亚洲淫性视频| 欧美午夜三级| 亚洲一品av免费观看| 久久se精品一区精品二区| 欧美激情按摩| 亚洲精品欧美日韩| 欧美中在线观看| 久久综合狠狠综合久久综青草 | 欧美日韩一区二区三区免费| 国产午夜精品美女视频明星a级| 亚洲午夜一区二区| 久久精品国产77777蜜臀| 韩国精品久久久999| 亚洲国产综合在线| 欧美大片免费观看| 国产麻豆午夜三级精品| 欧美亚洲免费| 欧美成人乱码一区二区三区| 亚洲日本成人| 午夜精品久久久久久久99樱桃| 国产日产精品一区二区三区四区的观看方式| 性色av一区二区三区| 欧美电影免费观看网站| 亚洲视频专区在线| 久久婷婷色综合| 亚洲区免费影片| 欧美亚洲一区三区| 亚洲第一区中文99精品| 亚洲一区免费观看| 樱桃视频在线观看一区| 亚洲一级二级| 国产一区香蕉久久| 99精品欧美一区二区蜜桃免费| 国产精品入口尤物| 亚洲深夜福利网站| 久久久久久久综合日本| 亚洲人成在线观看| 久久国产精品一区二区三区四区| 亚洲电影av| 欧美在线999| 亚洲日本欧美在线| 久久精品导航| 日韩一级在线观看| 久久久久久综合网天天| 亚洲精选91| 久久精品国产v日韩v亚洲| 亚洲日本视频| 久久超碰97人人做人人爱| 亚洲激情在线观看| 久久激情一区| 一本色道久久精品| 欧美91大片| 性欧美8khd高清极品| 欧美人与禽猛交乱配视频| 久久福利影视| 国产精品久久久亚洲一区 | 国内精品久久久久国产盗摄免费观看完整版 | 亚洲精品乱码久久久久久蜜桃91 | 亚洲美女在线一区| 麻豆精品视频在线观看视频| 亚洲资源在线观看| 欧美三级视频在线| 亚洲精品资源|