怎樣理解JavaSwing開發(fā)中的線程安全

這期內(nèi)容當中小編將會給大家?guī)碛嘘P(guān)怎樣理解Java Swing開發(fā)中的線程安全,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。

創(chuàng)新互聯(lián)建站是一家專業(yè)提供江川企業(yè)網(wǎng)站建設(shè),專注與網(wǎng)站制作、成都做網(wǎng)站、HTML5建站、小程序制作等業(yè)務(wù)。10年已為江川眾多企業(yè)、政府機構(gòu)等服務(wù)。創(chuàng)新互聯(lián)專業(yè)網(wǎng)站建設(shè)公司優(yōu)惠進行中。

SwingAPI的設(shè)計目標是強大、靈活和易用。非凡地,我們希望能讓程序員們方便地建立新的Swing組件,不論是從頭開始還是通過擴展我們所提供的一些組件。出于這個目的,我們不要求Swing組件支持多線程訪問。相反,我們向組件發(fā)送請求并在單一線程中執(zhí)行請求。

這里討論線程和Swing組件。目的不僅是為了幫助你以線程安全的方式使用SwingAPI,而且解釋了我們?yōu)槭裁磿x擇現(xiàn)在這樣的線程方案。

內(nèi)容:

單線程規(guī)則:Swing線程在同一時刻僅能被一個線程所訪問。一般來說,這個線程是事件派發(fā)線程。規(guī)則的例外:有些操作保證是線程安全的。事件分發(fā):假如你需要從事件處理或繪制代碼以外的地方訪問UI,那么你可以使用SwingUtilities類的invokeLater要求在事件派發(fā)線程中執(zhí)行某些代碼。這個方法會立即返回,不會等待代碼執(zhí)行完畢。invokeAndWait行為與invokeLater類似,除了這個方法會等待代碼執(zhí)行完畢。一般地,你可以用invokeLater來代替這個方法。下面是一些使用這幾個API的例子。尤其是以下幾個類:CardWindow、ControlPane、Player和OverallStatusPane。

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

RunnabledoWorkRunnable=newRunnable

};

SwingUtilities.invokeLater;使用invokeAndWait方法invokeAndWait方法和invokeLater方法很相似,除了invokeAndWait方法會等事件派發(fā)線程執(zhí)行了指定代碼才返回。在可能的情況下,你應(yīng)該盡量用invokeLater來代替invokeAndWait。假如你真的要使用invokeAndWait,請確保調(diào)用invokeAndWait的線程不會在調(diào)用期間持有任何其他線程可能需要的鎖。

這是一個使用invokeAndWait的例子:

voidshowHelloThereDialogthrowsException

};

SwingUtilities.invokeAndWait;

}

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

voidprintTextField

throwsException

};

SwingUtilities.invokeAndWait;

System.out.println;}

假如你能避免使用線程,***這樣做。線程可能難于使用,并使得程序的debug更困難。一般來說,對于嚴格意義下的GUI工作,線程是不必要的,比如對組件屬性的更新。不管怎么說,有時候線程是必要的。下列情況是使用線程的一些典型情況:執(zhí)行一項費時的任務(wù)而不必將事件派發(fā)線程鎖定。例子包括執(zhí)行大量計算的情況,會導致大量類被裝載的情況,和為網(wǎng)絡(luò)或磁盤I/O而阻塞的情況。重復地執(zhí)行一項操作,通常在兩次操作間間隔一個預(yù)定的時間周期。要等待來自客戶的消息。你可以使用兩個類來幫助你實現(xiàn)線程:SwingWorker:創(chuàng)建一個后臺線程來執(zhí)行費時的操作。Timer:創(chuàng)建一個線程來執(zhí)行或多次執(zhí)行某些代碼,在兩次執(zhí)行間間隔用戶定義的延遲。使用SwingWorker類SwingWorker類在SwingWorker.java中實現(xiàn),這個類并不包含在Java的任何發(fā)行版中,所以你必須單獨下載它。SwingWorker類做了所有實現(xiàn)一個后臺線程所需的骯臟工作。雖然許多程序都不需要后臺線程,后臺線程在執(zhí)行費時的操作時仍然是很有用的,它能提高程序的性能觀感。

SwingWorkersanexampleofusingSwingWorker:要使用SwingWorker類,你首先要實現(xiàn)它的一個子類。在子類中,你必須實現(xiàn)construct方法還包含你的長時間操作。當你實例化SwingWorker的子類時,SwingWorker創(chuàng)建一個線程但并不啟動它。你要調(diào)用你的SwingWorker對象的start方法來啟動線程,然后start方法會調(diào)用你的construct方法。當你需要construct方法返回的對象時,可以調(diào)用SwingWorker類的get方法。這是一個使用SwingWorker類的例子:

...//在main方法中:

finalSwingWorkerworker=newSwingWorker

};

worker.start;

...

//在動作事件處理方法中:

JOptionPane.showMessageDialog)

當程序的main方法調(diào)用start方法,SwingWorker啟動一個新的線程來實例化ExpensiveDialogComponent。main方法還構(gòu)造了由一個窗口和一個按鈕組成的GUI。當用戶點擊按鈕,程序?qū)⒆枞?,假如必要,阻塞到ExpensiveDialogComponent創(chuàng)建完成。然后程序顯示一個包含ExpensiveDialogComponent的模式對話框。你可以在MyApplication.java找到整個程序。使用Timer類Timer類通過一個ActionListener來執(zhí)行或多次執(zhí)行一項操作。你創(chuàng)建定時器的時候可以指定操作執(zhí)行的頻率,并且你可以指定定時器的動作事件的監(jiān)聽者。啟動定時器后,動作監(jiān)聽者的actionPerformed方法會被調(diào)用來執(zhí)行操作。定時器動作監(jiān)聽者定義的actionPerformed方法將在事件派發(fā)線程中調(diào)用。這意味著你不必在其中使用invokeLater方法。這是一個使用Timer類來實現(xiàn)動畫循環(huán)的例子:

publicclassAnimatorApplicationTimer

extendsJFrameimplementsActionListener

publicvoidstartAnimationelse

}

publicvoidstopAnimation

publicvoidactionPerformed

...

}

在一個線程中執(zhí)行所有的用戶界面代碼有這樣一些優(yōu)點:組件開發(fā)者不必對線程編程有深入的理解:像ViewPoint和Trestle這類工具包中的所有組件都必須完全支持多線程訪問,使得擴展非常困難,尤其對不精通線程編程的開發(fā)者來說。最近的一些工具包如SubArctic和IFC,都采用和Swing類似的設(shè)計。事件以可預(yù)知的次序派發(fā):invokeLater排隊的runnable對象從鼠標和鍵盤事件、定時器事件、繪制請求的同一個隊列派發(fā)。在一些組件完全支持多線程訪問的工具包中,組件的改變被變化無常的線程調(diào)度程序穿插到事件處理過程中。這使得全面測試變得困難甚至不可能。更低的代價:嘗試小心鎖住臨界區(qū)的工具包要花費實足的時間和空間在鎖的治理上。每當工具包中調(diào)用某個可能在客戶代碼中實現(xiàn)的方法時,工具包都要保存它的狀態(tài)并釋放所有鎖,以便客戶代碼能在必要時獲得鎖。當控制權(quán)交回到工具包,工具包又必須重新抓住它的鎖并恢復狀態(tài)。所有應(yīng)用程序都不得不負擔這一代價,即使大多數(shù)應(yīng)用程序并不需要對GUI的并發(fā)訪問。這是的SubArcticJavaToolkit的對在工具包中支持多線程訪問的問題的描述:我們的基本信條是,當設(shè)計和建造多線程應(yīng)用程序,尤其是那些包括GUI組件的應(yīng)用程序時,必須保證極端小心。線程的使用可能會很有欺騙性。在許多情況下,它們表現(xiàn)得能夠極好的簡化編成,使得設(shè)計“專注于單一任務(wù)的簡單自治實體”成為可能。在一些情況下它們的確簡化了設(shè)計和編碼。然而,在幾乎所有的情況下,它們都使得調(diào)試、測試和維護的困難大大增加甚至成為不可能。無論大多數(shù)程序員所受的練習、他們的經(jīng)驗和實踐,還是我們用來幫助自己的工具,都不是能夠用來對付非決定論的。例如,全面測試在bug依靠于時間時是幾乎不可能的。尤其對于Java來說,一個程序要運行在許多不同類型的機器的操作系統(tǒng)平臺上,并且每個程序都必須在搶先和非搶先式調(diào)度下都能正常工作。由于這些固有的困難,我們力勸你三思是否絕對有使用線程的必要。盡管如此,有些情況下使用線程是必要的,所以subArctic提供了一個線程安全的訪問機制。

上述就是小編為大家分享的怎樣理解Java Swing開發(fā)中的線程安全了,如果剛好有類似的疑惑,不妨參照上述分析進行理解。如果想知道更多相關(guān)知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。

當前題目:怎樣理解JavaSwing開發(fā)中的線程安全
轉(zhuǎn)載來源:http://muchs.cn/article32/jcpisc.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站導航、品牌網(wǎng)站設(shè)計、建站公司、企業(yè)網(wǎng)站制作做網(wǎng)站、品牌網(wǎng)站制作

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

商城網(wǎng)站建設(shè)