淺析Swing.SWT和AWT的區(qū)別
上一篇的比較主要是在API級(jí)別上的。讓我們將比較的焦點(diǎn)轉(zhuǎn)移到實(shí)現(xiàn)細(xì)節(jié)上。Swing.SWT和AWT的區(qū)別是Swing是純Java實(shí)現(xiàn),而SWT和AWT是Java和JNI的混合。當(dāng)然,它們的目標(biāo)都是相同的,提供一個(gè)跨平臺(tái)的APIs。然而為了達(dá)到這一點(diǎn),SWT和AWT不得不犧牲一些組件和特性以提供一個(gè)通用的APIs。
AWT
一個(gè)AWT組件通常是一個(gè)包含了對(duì)等體接口類型引用的組件類。這個(gè)引用指向本地對(duì)等體實(shí)現(xiàn)。舉java.awt.Label為例,它的對(duì)等體接口是 LabelPeer。LabelPeer是平臺(tái)無關(guān)的。在不同平臺(tái)上,AWT提供不同的對(duì)等體類來實(shí)現(xiàn)LabelPeer。在Windows上,對(duì)等體類是WlabelPeer,它調(diào)用JNI來實(shí)現(xiàn)label的功能。這些JNI方法用C或C++編寫。它們關(guān)聯(lián)一個(gè)本地的label,真正的行為都在這里發(fā)生。作為整體,AWT組件由AWT組件類和AWT對(duì)等體提供了一個(gè)全局公用的API給應(yīng)用程序使用。一個(gè)組件類和它的對(duì)等體接口是平臺(tái)無關(guān)的。底層的對(duì)等體類和JNI代碼是平臺(tái)相關(guān)的。
SWT
SWT也使用JNI的方法論來實(shí)現(xiàn)。但細(xì)節(jié)不同于AWT。SWT的擁護(hù)者聽到人們拿SWT和AWT相提并論可是會(huì)很生氣的,Steve Northover,SWT之父,就曾為此抱怨過。
沒錯(cuò),它們是不同的。讓我們深究SWT的代碼。在SWT中,各個(gè)平臺(tái)上唯一相同的部分是組件的接口,是類和方法的定義簽名。所有的底層代碼都是平臺(tái)差異的。 SWT為每個(gè)平臺(tái)提供了OS類。這個(gè)類用JNI封裝了許多本地APIs。SWT組件類通過把這些JNI方法黏合在一起提供一個(gè)有意義的功能。
例如,在Windows上,文本域的選擇是由一個(gè)系統(tǒng)調(diào)用處理的。這個(gè)系統(tǒng)調(diào)用在Windows的OS類中作為一個(gè)本地方法實(shí)現(xiàn)。所以在Windows平臺(tái)的Text的setSelection方法中只用到了一個(gè)JNI調(diào)用。
然而,在motif上,文本域的選擇包含兩個(gè)本地調(diào)用。SWT就在motif的OS類中實(shí)現(xiàn)了兩個(gè)調(diào)用。所以在motif上組件類需要作兩次調(diào)用來實(shí)現(xiàn)文本的選擇。
現(xiàn)在你應(yīng)該能看出SWT和AWT的***不同了,它們使用了不同的對(duì)等體編程方式來消除平臺(tái)差異。SWT用java代碼或有JNI實(shí)現(xiàn)的java對(duì)等體來黏合系統(tǒng)調(diào)用。而AWT把代碼包含在對(duì)等體中,使情況復(fù)雜化了,我個(gè)人覺得SWT的方法更加明智。[是否我翻譯有問題,因?yàn)槲也⒉挥X得是這樣更明智,SWT 的無則模擬是不必要的,這是使用者才去做的事,SWT作為提供者應(yīng)該無則C++實(shí)現(xiàn),當(dāng)然實(shí)現(xiàn)的是最核心的高度復(fù)用的又或者需要極大性能支持的,畢竟帶了動(dòng)態(tài)鏈接庫(kù),索性多放點(diǎn)東西。
Swing
到了Swing這里,一切就變得清晰和直接了。除了頂層容器,Swing的實(shí)現(xiàn)不依賴于具體平臺(tái)。它掌管了所有的控制和資源。Swing所需要的是事件輸入來驅(qū)動(dòng)系統(tǒng),以及承接自頂層AWT容器的圖形處理,字體和顏色。普通的Swing組件可以看作是AWT容器的一塊邏輯區(qū)域。它們并沒有注冊(cè)對(duì)等體。所有添加到同一頂層容器的Swing組件共享它的AWT對(duì)等體以獲取系統(tǒng)資源,如字體,圖形處理等。Swing將組件自己的數(shù)據(jù)結(jié)構(gòu)存儲(chǔ)在JVM的空間中。它完全由自己管理畫圖處理,事件分發(fā)和組件布局。
【編輯推薦】