web-dev-qa-db-ja.com

コードからJava Swingアプリケーションを閉じる方法

コードからSwingアプリケーションを終了する適切な方法は何ですか?また、落とし穴は何ですか?

タイマーが切れたら、アプリケーションを自動的に閉じようとしました。しかし、JFramedispose()を呼び出すだけでは、トリックは行われませんでした。ウィンドウは消えましたが、アプリケーションは終了しませんでした。ただし、閉じるボタンでウィンドウを閉じると、アプリケーションは終了します。私は何をすべきか?

89

JFrameのデフォルトのクローズアクションは、DISPOSE_ON_CLOSEの代わりに "EXIT_ON_CLOSE"に設定できます(EXIT_ON_CLOSEを使い続ける理由は私を超えています)。

破棄されていないウィンドウまたはデーモン以外のスレッドがある場合、アプリケーションは終了しません。これはエラーと見なされる必要があります(System.exitで解決するのは非常に悪い考えです)。

最も一般的な原因は、Java.util.Timerと作成したカスタムスレッドです。両方をデーモンに設定するか、明示的に強制終了する必要があります。

すべてのアクティブなフレームを確認する場合は、 Frame.getFrames() を使用できます。すべてのWindows/Frameが破棄される場合は、デバッガーを使用して、まだ実行中の非デーモンスレッドがないか確認します。

103
James Schek

EXIT_ON_CLOSE

frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

Window Listener を記述して、実際にアプリを終了する前にいくつかのクリーニング操作を行うことができるため、System.exit(0)の前が優れています。

そのウィンドウリスナーを使用すると、以下を定義できます。

public void windowClosing(WindowEvent e) {
    displayMessage("WindowListener method called: windowClosing.");
    //A pause so user can see the message before
    //the window actually closes.
    ActionListener task = new ActionListener() {
        boolean alreadyDisposed = false;
        public void actionPerformed(ActionEvent e) {
            if (frame.isDisplayable()) {
                alreadyDisposed = true;
                frame.dispose();
            }
        }
    };
    Timer timer = new Timer(500, task); //fire every half second
    timer.setInitialDelay(2000);        //first delay 2 seconds
    timer.setRepeats(false);
    timer.start();
}

public void windowClosed(WindowEvent e) {
    //This will only be seen on standard output.
    displayMessage("WindowListener method called: windowClosed.");
}
34
VonC

試してください:

System.exit(0);

粗野ですが、効果的です。

15
Daniel Spiewak

安全な方法かもしれません:

    private JButton btnExit;
    ...
    btnExit = new JButton("Quit");      
    btnExit.addActionListener(new ActionListener() {
        public void actionPerformed(ActionEvent e){
            Container frame = btnExit.getParent();
            do 
                frame = frame.getParent(); 
            while (!(frame instanceof JFrame));                                      
            ((JFrame) frame).dispose();
        }
    });
11
Kachwahed

次のプログラムには、System.exit()を明示的に呼び出さずに、無関係なスレッドのないプログラムを終了するコードが含まれています。スレッド/リスナー/タイマーなどを使用するアプリケーションにこの例を適用するには、actionPerformed()内でWindowEventを手動で開始する前に、終了を要求する(必要に応じて待機する)クリーンアップコードを挿入するだけです。

示されているとおりに正確に実行できるコードをコピー/貼り付けする場合は、最後に少しugいが、それ以外は無関係なmainメソッドが含まれます。

public class CloseExample extends JFrame implements ActionListener {

    private JButton turnOffButton;

    private void addStuff() {
        setDefaultCloseOperation(DISPOSE_ON_CLOSE);
        turnOffButton = new JButton("Exit");
        turnOffButton.addActionListener(this);
        this.add(turnOffButton);
    }

    public void actionPerformed(ActionEvent quitEvent) {
        /* Iterate through and close all timers, threads, etc here */
        this.processWindowEvent(
                new WindowEvent(
                      this, WindowEvent.WINDOW_CLOSING));
    }

    public CloseExample() {
        super("Close Me!");
        addStuff();
    }

    public static void main(String[] args) {
        Java.awt.EventQueue.invokeLater(new Runnable() {
            public void run() {
                CloseExample cTW = new CloseExample();
                cTW.setSize(200, 100);
                cTW.setLocation(300,300);
                cTW.setVisible(true);
            }
        });
    }
}
9
Eric Sadler

私があなたを正しく理解していれば、ユーザーが閉じるボタンをクリックしなくてもアプリケーションを閉じたいと思うでしょう。 WindowEventsは、おそらくaddWindowListener()またはenableEvents()のうち、ニーズに合った方で登録する必要があります。

その後、processWindowEvent()を呼び出してイベントを呼び出すことができます。 JFrameを作成し、5秒待機して、ユーザーの操作なしでJFrameを閉じるサンプルコードを次に示します。

import javax.swing.*;
import Java.awt.*;
import Java.awt.event.*;

public class ClosingFrame extends JFrame implements WindowListener{

public ClosingFrame(){
    super("A Frame");
    setSize(400, 400);
            //in case the user closes the window
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
            setVisible(true);
            //enables Window Events on this Component
            this.addWindowListener(this);

            //start a timer
    Thread t = new Timer();
            t.start();
    }

public void windowOpened(WindowEvent e){}
public void windowClosing(WindowEvent e){}

    //the event that we are interested in
public void windowClosed(WindowEvent e){
    System.exit(0);
}

public void windowIconified(WindowEvent e){}
public void windowDeiconified(WindowEvent e){}
public void windowActivated(WindowEvent e){}
public void windowDeactivated(WindowEvent e){}

    //a simple timer 
    class Timer extends Thread{
           int time = 10;
           public void run(){
     while(time-- > 0){
       System.out.println("Still Waiting:" + time);
               try{
                 sleep(500);                     
               }catch(InterruptedException e){}
             }
             System.out.println("About to close");
    //close the frame
            ClosingFrame.this.processWindowEvent(
                 new WindowEvent(
                       ClosingFrame.this, WindowEvent.WINDOW_CLOSED));
           }
    }

    //instantiate the Frame
public static void main(String args[]){
          new ClosingFrame();
    }

}

ご覧のように、processWindowEvent()メソッドは、アプリケーションを閉じる前に必要な場合にクリーンアップコードを実行する機会がある場合にWindowClosedイベントを発生させます。

4

Oracle Documentation をご覧ください。

JDK 1.4以降、アプリケーションは次の場合に終了します。

  • 表示可能なAWTまたはSwingコンポーネントはありません。
  • ネイティブイベントキューにネイティブイベントはありません。
  • Java EventQueuesにはAWTイベントはありません。

コーナーケース:

ドキュメントには、一部のパッケージが表示可能なコンポーネントをリリースせずに作成することが記載されています。 Toolkit.getDefaultToolkit()を呼び出すプログラムは終了しません。 例。

また、他のプロセスは、何らかの理由でイベントをネイティブイベントキューに送信しているときにAWTを存続させることができます。

また、一部のシステムでは、アプリケーションが実際に終了するまでに数秒かかることに気付きました。

2
Reiner Lange

他のコメントに応じて、DISPOSE_ON_CLOSEはアプリケーションを適切に終了しないようです。ウィンドウを破壊するだけですが、アプリケーションは実行を継続します。アプリケーションを終了する場合は、EXIT_ON_CLOSEを使用します。

0
JSideris

考えはここにWindowListenerがあると思う-物事がシャットダウンする前に実行したい任意のコードをそこに追加できる

0
Tamas Rev