線程與Swing

作者:Hans Muller,Kathy Walrath 
翻譯:郭曉剛([email protected]
原文來自java.sun.com

本文關於Swing中的多線程,發表於1998年4月。一個月後,我們發表了另一篇文章《使用Swing Worker線程》,該文更深入地討論了這一主題。要更好地瞭解多線程在Swing中如何工作,我們建議你把這兩篇文章都看一下。
注意:在2000年9月我們修改了這篇文章和它的例子以適用於一個更新版本的SwingWorker類。SwingWorker類的這個版本修正了一些微妙的線程bug。

Swing API的設計目標是強大、靈活和易用。特別地,我們希望能讓程序員們方便地建立新的Swing組件,不論是從頭開始還是通過擴展我們所提供的一些組件。
出於這個目的,我們不要求Swing組件支持多線程訪問。相反,我們向組件發送請求並在單一線程中執行請求。
本文討論線程和Swing組件。目的不僅是爲了幫助你以線程安全的方式使用Swing API,而且解釋了我們爲什麼會選擇現在這樣的線程方案。
本文包括以下內容:
  • 單線程規則:Swing線程在同一時刻僅能被一個線程所訪問。一般來說,這個線程是事件派發線程(event-dispatching thread)。
  • 規則的例外:有些操作保證是線程安全的。
  • 事件分發:如果你需要從事件處理(event-handling)或繪製代碼以外的地方訪問UI,那麼你可以使用SwingUtilities類的invokeLater()或invokeAndWait()方法。
  • 創建線程:如果你需要創建一個線程——比如用來處理一些耗費大量計算能力或受I/O能力限制的工作——你可以使用一個線程工具類如SwingWorker或Timer。
  • 爲什麼我們這樣實現Swing:我們用一些關於Swing的線程安全的背景資料來結束這篇文章。

Swing的規則是:
一旦Swing組件被具現化(realized),所有可能影響或依賴於組件狀態的代碼都應該在事件派發線程中執行。

這個規則可能聽起來有點嚇人,但對許多簡單的程序來說,你用不着爲線程問題操心。在我們深入如何撰寫Swing代碼之前,讓我們先來定義兩個術語:具現化(realized)和事件派發線程(event-dispatching thread)。
具現化的意思是組建的paint()方法已經或可能會被調用。一個作爲頂級窗口的Swing組件當調用以下方法時將被具現化:setVisible(true)、show()或(可能令你驚奇)pack()。當一個窗口被具現化,它包含的所有組件都被具現化。另一個具現化一個組件的方法是將它放入到一個已經具現化的容器中。稍後你會看到一些對組件具現化的例子。
事件派發線程是執行繪製和事件處理的線程。例如,paint()和actionPerformed()方法會自動在事件派發線程中執行。另一個將代碼放到事件派發線程中執行的方法是使用SwingUtilities類的invokeLater()方法。
所有可能影響一個已具現化的Swing組件的代碼都必須在事件派發線程中執行。但這個規則有一些例外:
  • 有些方法是線程安全的:在Swing API的文檔中,線程安全的方法用以下文字標記:
    This method is thread safe, although most Swing methods are not. 
    (這個方法是線程安全的,儘管大多數Swing方法都不是。)
  • 一個應用程序的GUI常常可以在主線程中構建和顯示:下面的典型代碼是安全的,只要沒有(Swing或其他)組件被具現化:

    public class MyApplication {
    public static void main(String[] args) {
       JFrame f = new JFrame("Labels");
       // 在這裏將各組件 
       // 加入到主框架…… 
       f.pack(); 
       f.show(); 
       // 不要再做任何GUI工作…… 
       } 
    }

上面所示的代碼全部在“main”線程中運行。對f.pack()的調用使得JFrame以下的組件都被具現化。這意味着,f.show()調用是不安全的且應該在事件派發線程中執行。儘管如此,只要程序還沒有一個看得到的GUI,JFrame或它的裏面的組件就幾乎不可能在f.show()返回前收到一個paint()調用。因爲在f.show()調用之後不再有任何GUI代碼,於是所有GUI工作都從主線程轉到了事件派發線程,因此前面所討論的代碼實際上是線程安全的。
  • 一個applet的GUI可以在init()方法中構造和顯示:現有的瀏覽器都不會在一個applet的init()和start()方法被調用前繪製它。因而,在一個applet的init()方法中構造GUI是安全的,只要你不對applet中的對象調用show()或setVisible(true)方法。
    要順便一提的是,如果applet中使用了Swing組件,就必須實現爲JApplet的子類。並且,組件應該添加到的JApplet內容窗格(content pane)中,而不要直接添加到JApplet。對任何applet,你都不應該在init()或start()方法中執行費時的初始化操作;而應該啓動一個線程來執行費時的任務。
  • 下述JComponent方法是安全的,可以從任何線程調用:repaint()、revalidate()、和invalidate()。repaint()和revalidate()方法爲事件派發線程對請求排隊,並分別調用paint()和validate()方法。invalidate()方法只在需要確認時標記一個組件和它的所有直接祖先。
  • 監聽者列表可以由任何線程修改:調用addListenerTypeListener()和removeListenerTypeListener()方法總是安全的。對監聽者列表的添加/刪除操作不會對進行中的事件派發有任何影響。

注意:revalidate()和舊的validate()方法之間的重要區別是,revalidate()會緩存請求並組合成一次validate()調用。這和repaint()緩存並組合繪製請求類似。
大多數初始化後的GUI工作自然地發生在事件派發線程。一旦GUI成爲可見,大多數程序都是由事件驅動的,如按鈕動作或鼠標點擊,這些總是在事件派發線程中處理的。
不過,總有些程序需要在GUI成爲可見後執行一些非事件驅動的GUI工作。比如:
  • 在成爲可用前需要進行長時間初始化操作的程序:這類程序通常應該在初始化期間就顯示出GUI,然後更新或改變GUI。初始化過程不應該在事件派發線程中進行;否則,重繪組件和事件派發會停止。儘管如此,在初始化之後,GUI的更新/改變還是應該在事件派發線程中進行,理由是線程安全。
  • 必須響應非AWT事件來更新GUI的程序:例如,想象一個服務器程序從可能運行在其他機器上的程序得到請求。這些請求可能在任何時刻到達,並且會引起在一些可能未知的線程中對服務器的方法調用。這個方法調用怎樣更新GUI呢?在事件派發線程中執行GUI更新代碼。

SwingUtilities類提供了兩個方法來幫助你在事件派發線程中執行代碼:
  • invokeLater():要求在事件派發線程中執行某些代碼。這個方法會立即返回,不會等待代碼執行完畢。
  • invokeAndWait():行爲與invokeLater()類似,除了這個方法會等待代碼執行完畢。一般地,你可以用invokeLater()來代替這個方法。

下面是一些使用這幾個API的例子。請同時參閱《The Java Tutorial》中的“BINGO example”,尤其是以下幾個類:CardWindowControlPanePlayerOverallStatusPane

使用invokeLater()方法


你可以從任何線程調用invokeLater()方法以請求事件派發線程運行特定代碼。你必須把要運行的代碼放到一個Runnable對象的run()方法中,並將此Runnable對象設爲invokeLater()的參數。invokeLater()方法會立即返回,不等待事件派發線程執行指定代碼。這是一個使用invokeLater()方法的例子:

Runnable doWorkRunnable = new Runnable() {
    public void run() { doWork(); }
};
SwingUtilities.invokeLater(doWorkRunnable);

使用invokeAndWait()方法


invokeAndWait()方法和invokeLater()方法很相似,除了invokeAndWait()方法會等事件派發線程執行了指定代碼才返回。在可能的情況下,你應該儘量用invokeLater()來代替invokeAndWait()。如果你真的要使用invokeAndWait(),請確保調用invokeAndWait()的線程不會在調用期間持有任何其他線程可能需要的鎖。
這是一個使用invokeAndWait()的例子:

void showHelloThereDialog() 
        throws Exception {
    Runnable showModalDialog = new 
      Runnable() {
        public void run() {
            JOptionPane.showMessageDialog(
               myMainFrame, "Hello There");
        }
    };
    SwingUtilities.invokeAndWait
       (showModalDialog);
}

類似地,假設一個線程需要對GUI的狀態進行存取,比如文本域的內容,它的代碼可能類似這樣:

void printTextField() throws Exception {
    final String[] myStrings = 
       new String[2];

    Runnable getTextFieldText = 
      new Runnable() {
        public void run() {
            myStrings[0] = 
               textField0.getText();
            myStrings[1] = 
               textField1.getText();
        }
    };
    SwingUtilities.invokeAndWait
      (getTextFieldText);

    System.out.println(myStrings[0] 
                       + " " + myStrings[1]);
}

如果你能避免使用線程,最好這樣做。線程可能難於使用,並使得程序的debug更困難。一般來說,對於嚴格意義下的GUI工作,線程是不必要的,比如對組件屬性的更新。
不管怎麼說,有時候線程是必要的。下列情況是使用線程的一些典型情況:
  • 執行一項費時的任務而不必將事件派發線程鎖定。例子包括執行大量計算的情況,會導致大量類被裝載的情況(如初始化),和爲網絡或磁盤I/O而阻塞的情況。
  • 重複地執行一項操作,通常在兩次操作間間隔一個預定的時間週期。
  • 要等待來自客戶的消息。

你可以使用兩個類來幫助你實現線程:
  • SwingWorker:創建一個後臺線程來執行費時的操作。
  • Timer:創建一個線程來執行或多次執行某些代碼,在兩次執行間間隔用戶定義的延遲。

使用SwingWorker類


SwingWorker類在SwingWorker.java中實現,這個類並不包含在Java的任何發行版中,所以你必須單獨下載它。
SwingWorker類做了所有實現一個後臺線程所需的骯髒工作。雖然許多程序都不需要後臺線程,後臺線程在執行費時的操作時仍然是很有用的,它能提高程序的性能觀感。
SwingWorker's get() method. Here's an example of using SwingWorker:
要使用SwingWorker類,你首先要實現它的一個子類。在子類中,你必須實現construct()方法還包含你的長時間操作。當你實例化SwingWorker的子類時,SwingWorker創建一個線程但並不啓動它。你要調用你的SwingWorker對象的start()方法來啓動線程,然後start()方法會調用你的construct()方法。當你需要construct()方法返回的對象時,可以調用SwingWorker類的get()方法。這是一個使用SwingWorker類的例子:

...// 在main方法中:
    final SwingWorker worker = 
      new SwingWorker() {
        public Object construct() {
            return new 
               expensiveDialogComponent();
        }
    };
    worker.start();

...// 在動作事件處理方法中:
    JOptionPane.showMessageDialog
        (f, worker.get());

當程序的main()方法調用start()方法,SwingWorker啓動一個新的線程來實例化ExpensiveDialogComponent。main()方法還構造了由一個窗口和一個按鈕組成的GUI。
當用戶點擊按鈕,程序將阻塞,如果必要,阻塞到ExpensiveDialogComponent創建完成。然後程序顯示一個包含ExpensiveDialogComponent的模式對話框。你可以在MyApplication.java找到整個程序。

使用Timer類


Timer類通過一個ActionListener來執行或多次執行一項操作。你創建定時器的時候可以指定操作執行的頻率,並且你可以指定定時器的動作事件的監聽者(action listener)。啓動定時器後,動作監聽者的actionPerformed()方法會被(多次)調用來執行操作。
定時器動作監聽者(action listener)定義的actionPerformed()方法將在事件派發線程中調用。這意味着你不必在其中使用invokeLater()方法。
這是一個使用Timer類來實現動畫循環的例子:

public class AnimatorApplicationTimer 
  extends JFrame implements 
  ActionListener {
    ...//在這裏定義實例變量
    Timer timer;

    public AnimatorApplicationTimer(...) {
        ...
        // 創建一個定時器來  
        // 來調用此對象action handler。
        timer = new Timer(delay, this);
        timer.setInitialDelay(0);
        timer.setCoalesce(true);
        ...
    }

    public void startAnimation() {
        if (frozen) {
            // 什麼都不做。應用戶要求 
            // 停止變換圖像。
        } else {
            // 啓動(或重啓動)動畫!
            timer.start();
        }
    }

    public void stopAnimation() {
        // 停止動畫線程。
        timer.stop();
    }

    public void actionPerformed
      (ActionEvent e) {
        // 進到下一幀動畫。
        frameNumber++;

        // 顯示。
        repaint();
    }
    ...
}

在一個線程中執行所有的用戶界面代碼有這樣一些優點:
  • 組件開發者不必對線程編程有深入的理解:像ViewPoint和Trestle這類工具包中的所有組件都必須完全支持多線程訪問,使得擴展非常困難,尤其對不精通線程編程的開發者來說。最近的一些工具包如SubArctic和IFC,都採用和Swing類似的設計。
  • 事件以可預知的次序派發:invokeLater()排隊的runnable對象從鼠標和鍵盤事件、定時器事件、繪製請求的同一個隊列派發。在一些組件完全支持多線程訪問的工具包中,組件的改變被變化無常的線程調度程序穿插到事件處理過程中。這使得全面測試變得困難甚至不可能。
  • 更低的代價:嘗試小心鎖住臨界區的工具包要花費實足的時間和空間在鎖的管理上。每當工具包中調用某個可能在客戶代碼中實現的方法時(如public類中的任何public和protected方法),工具包都要保存它的狀態並釋放所有鎖,以便客戶代碼能在必要時獲得鎖。當控制權交回到工具包,工具包又必須重新抓住它的鎖並恢復狀態。所有應用程序都不得不負擔這一代價,即使大多數應用程序並不需要對GUI的併發訪問。

這是的SubArctic Java Toolkit的作者對在工具包中支持多線程訪問的問題的描述:
我們的基本信條是,當設計和建造多線程應用程序,尤其是那些包括GUI組件的應用程序時,必須保證極端小心。線程的使用可能會很有欺騙性。在許多情況下,它們表現得能夠極好的簡化編成,使得設計“專注於單一任務的簡單自治實體”成爲可能。在一些情況下它們的確簡化了設計和編碼。然而,在幾乎所有的情況下,它們都使得調試、測試和維護的困難大大增加甚至成爲不可能。無論大多數程序員所受的訓練、他們的經驗和實踐,還是我們用來幫助自己的工具,都不是能夠用來對付非決定論的。例如,全面測試(這總是困難的)在bug依賴於時間時是幾乎不可能的。尤其對於Java來說,一個程序要運行在許多不同類型的機器的操作系統平臺上,並且每個程序都必須在搶先和非搶先式調度下都能正常工作。
由於這些固有的困難,我們力勸你三思是否絕對有使用線程的必要。儘管如此,有些情況下使用線程是必要的(或者是被其他軟件包強加的),所以subArctic提供了一個線程安全的訪問機制。本章討論了這一機制和怎樣在一個獨立線程中安全地操作交互樹。
他們所說的線程安全機制非常類似於SwingUtilities類提供的invokeLater()和invokeAndWait()方法。
 
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章