Java GUI应用程序关闭陷阱
生活随笔
收集整理的這篇文章主要介紹了
Java GUI应用程序关闭陷阱
小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.
最近,我遇到了一個(gè)或兩個(gè)Java GUI應(yīng)用程序在關(guān)閉時(shí)無法關(guān)閉的問題。 它們似乎是一個(gè)過程,消耗著計(jì)算機(jī)資源。 今天,我深入探究了問題的根源,這是一個(gè)我以前從未意識(shí)到的棘手問題,所以我想我會(huì)分享一下。 理論上,當(dāng)您關(guān)閉Java應(yīng)用程序時(shí),應(yīng)停止所有線程,并且進(jìn)程應(yīng)終止。 就我而言,當(dāng)我監(jiān)視應(yīng)用程序時(shí),我希望完成的線程(例如Swing工作池)仍然處于活動(dòng)狀態(tài),即Strange。 原因是AWT Shutdown線程沒有終止所有幫助程序線程,原因是EventQueues中仍然存在AWT事件。 我將解釋其原因是真正的偷偷摸摸的小gatcha。 我的應(yīng)用程序使用了一個(gè)具有常規(guī)睡眠的線程,但是當(dāng)醒來時(shí)會(huì)進(jìn)行一些計(jì)算,然后調(diào)用以更新gui: Thread updateThread = new Thread(new Runnable() {@Override
public void run() {int i = 0;do {try {Thread.sleep(300); // 300msgui.updateValue(SOME_VALUE)} catch(InterruptException ex) {return;} frame.setValue(SOMEDATA); } while(i++ < 100); } }, "updateThread");updateThread.setDaemon(true);
updateThread.start(); 現(xiàn)在您將注意到,如果線程被中斷并且作為守護(hù)程序線程啟動(dòng),則該線程返回。 我曾以為,作為應(yīng)用程序關(guān)閉的一部分,線程將終止,但實(shí)際上不是。 這是由gui.updateValue(SOME_VALUE)使用InvokeLater引起的: public void updateValue(final int value) {// make sure we access graphics in the EDT threadjava.awt.EventQueue.invokeLater(new Runnable() {@Override public void run() { try { ......... SOME CODE }catch(Exception t) {// not a lot to do } } });
} InvokeLater基本上是在EventQueue上放置一個(gè)事件,因此AWT Shutdown線程想要關(guān)閉應(yīng)用程序。 AWT Shutdown線程每秒鐘檢查一次EventQueues,但是正如您將看到的,我的Thread會(huì)執(zhí)行更新一秒(300ms),因此隊(duì)列上始終有一個(gè)事件! 簡(jiǎn)而言之,AWT Shutdown線程永遠(yuǎn)不會(huì)終止我希望其終止的線程,因此需要終止應(yīng)用程序。 在我的線程的while循環(huán)中,變通方法很簡(jiǎn)單,我還檢查了通過它進(jìn)行更新的JComonent是否仍然可見并顯示,如果不是退出循環(huán),則該線程死亡,因此沒有其他事件放在事件線程上,應(yīng)用程序按預(yù)期關(guān)閉:) Thread updateThread = new Thread(new Runnable() {@Override
public void run() {int i = 0; do { try { Thread.sleep(300); // 300msgui.updateValue(SOME_VALUE); }catch(InterruptException ex) { return;} frame.setValue(SOMEDATA);}while(i < 100 && progressGlassPane.isVisible() && progressGlassPane.isShowing());}
}, "updateThread");
updateThread.setDaemon(true);
updateThread.start(); 因此,簡(jiǎn)而言之,不要以低于一秒的頻率從幫助程序線程中調(diào)用InvokeLater,除非在正在更新的組件不再可見的情況下也終止了該線程! 作為旁注,發(fā)現(xiàn)問題后,我發(fā)現(xiàn)這非常
參考: Coal Face博客上Java桌面開發(fā)的 JCG合作伙伴 Steve Webb的Java GUI Application Shutdown Gotcha 。
翻譯自: https://www.javacodegeeks.com/2012/05/java-gui-application-shutdown-gotcha.html
總結(jié)
以上是生活随笔為你收集整理的Java GUI应用程序关闭陷阱的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 欧洲航天局推迟阿丽亚娜 6 火箭热火测试
- 下一篇: Java陷阱:内部类中的字段访问