Swing線程解決方案
不正確的Swing線程是運行緩慢、無響應(yīng)和不穩(wěn)定的Swing應(yīng)用的主要原因之一。這是許多原因造成的,從開發(fā)人員對Swing單線程模型的誤解,到保證正確的線程執(zhí)行的困難。即使對Swing線程進(jìn)行了很多努力,應(yīng)用線程邏輯也是很難理解和維護(hù)的。本文闡述了如何在開發(fā)Swing應(yīng)用中使用事件驅(qū)動編程,以大大簡化開發(fā)、維護(hù),并提供高靈活性。
既然我們是要簡化Swing應(yīng)用的線程,首先讓我們來看看 Swing線程是怎么工作的,為什么它是必須的。Swing API是圍繞單線程模型設(shè)計的。這意味著Swing組件必須總是通過同一個線程來修改和操縱。為什么采用單線程模型,這有很多原因,包括開發(fā)成本和同步 Swing的復(fù)雜性--這都會造成一個遲鈍的API。為了達(dá)到單線程模型,有一個專門的線程用于和Swing組件交互。這個線程就是大家熟知的Swing線程,AWT(有時也發(fā)音為“ought”)線程,或者事件分派線程。在本文的下面的部分,我選用Swing線程的叫法。
既然Swing線程是和 Swing組件進(jìn)行交互的唯一的線程,它就被賦予了很多責(zé)任。所有的繪制和圖形,鼠標(biāo)事件,組件事件,按鈕事件,和所有其它事件都發(fā)生在Swing線程。因為Swing線程的工作已經(jīng)非常沉重了,當(dāng)太多其它工作在Swing線程中進(jìn)行處理時就會發(fā)生問題。會引起這個問題的最常見的位置是在非Swing處理的地方,像發(fā)生在一個事件監(jiān)聽器方法中,比如JButton的ActionListener,的數(shù)據(jù)庫查找。既然ActionListener的 actionPerformed()方法自動在Swing線程中執(zhí)行,那么,數(shù)據(jù)庫查找也將在Swing線程中執(zhí)行。這將占用了Swing的工作,阻止它處理它的其它任務(wù)--像繪制,響應(yīng)鼠標(biāo)移動,處理按鈕事件,和應(yīng)用的縮放。用戶以為應(yīng)用死掉了,但實際上并不是這樣。在適當(dāng)?shù)木€程中執(zhí)行代碼對確保系統(tǒng)正常地執(zhí)行非常重要。
既然我們已經(jīng)看到了在適當(dāng)?shù)木€程中執(zhí)行Swing應(yīng)用的代碼是多么重要,現(xiàn)在讓我們?nèi)绾螌崿F(xiàn)這些線程。我們看看將代碼放入和移出Swing線程的標(biāo)準(zhǔn)機制。在講述過程中,我將突出幾個和標(biāo)準(zhǔn)機制有關(guān)的問題和難點。正如我們看到的,大部分的問題都來自于企圖在異步的Swing線程模型上實現(xiàn)同步的代碼模型。從那兒,我們將看到如何修改我們的例子到事件驅(qū)動--移植整個方式到異步模型。
通用Swing線程解決方案
讓我們以一個最常用的Swing線程錯誤開始。我們將企圖使用標(biāo)準(zhǔn)的技術(shù)來修正這個問題。在這個過程中,我們將看到實現(xiàn)正確的Swing線程的復(fù)雜性和常見困難。并且,注意在修正這個Swing線程問題中,許多中間的例子也是不能工作的。在例子中,我在代碼失敗的地方以//broken開頭標(biāo)出。好了,現(xiàn)在,讓我們進(jìn)入我們的例子吧。
假設(shè)我們在執(zhí)行圖書查找。我們有一個簡單的用戶界面,包括一個查找文本域,一個查找按鈕,和一個輸出的文本區(qū)域。不要批評我的UI設(shè)計,這個確實很丑陋,我承認(rèn)。
用戶輸入書的標(biāo)題,作者或者其它條件,然后顯示一個結(jié)果的列表。下面的代碼例子演示了按鈕的ActionListener在同一個線程中調(diào)用 lookup()方法。在這些例子中,我使用了thread.sleep()休眠5秒來作為一個占位的外部查找。線程休眠的結(jié)果等同于一個耗時5秒的同步的服務(wù)器調(diào)用。
- private void searchButton_actionPerformed()
- {
- outputTA.setText("Searching for: " + searchTF.getText());
- //Broken!! Too much work in the Swing
- thread String[] results = lookup(searchTF.getText());
- outputTA.setText("");
- for (int i = 0; i < results.length; i++)
- {
- String result = results[i];
- outputTA.setText(outputTA.getText() + ´´ ´´ + result);
- }
- }
【編輯推薦】