Javaのシャットダウンフックの便利な例は?


125

私のJavaアプリケーションが堅牢になるために妥当な手順を実行することを確認しようとしています。その一環として、正常にシャットダウンする必要があります。私はシャットダウンフックについて読んでいますが、実際にはそれらをどのように使用するのかわかりません。

そこに実用的な例はありますか?

たとえば、次のような単純なアプリケーションで、数値をファイルに、10を1行に、100のバッチで書き込むとします。プログラムが中断された場合、特定のバッチが確実に終了するようにしたいとします。シャットダウンフックを登録する方法はわかりますが、シャットダウンフックをアプリケーションに統合する方法がわかりません。助言がありますか?

package com.example.test.concurrency;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.PrintWriter;

public class GracefulShutdownTest1 {
    final private int N;
    final private File f;
    public GracefulShutdownTest1(File f, int N) { this.f=f; this.N = N; }

    public void run()
    {
        PrintWriter pw = null;
        try {
            FileOutputStream fos = new FileOutputStream(this.f);
            pw = new PrintWriter(fos);
            for (int i = 0; i < N; ++i)
                writeBatch(pw, i);
        }
        catch (FileNotFoundException e) {
            e.printStackTrace();
        }
        finally
        {
            pw.close();
        }       
    }

    private void writeBatch(PrintWriter pw, int i) {
        for (int j = 0; j < 100; ++j)
        {
            int k = i*100+j;
            pw.write(Integer.toString(k));
            if ((j+1)%10 == 0)
                pw.write('\n');
            else
                pw.write(' ');
        }
    }

    static public void main(String[] args)
    {
        if (args.length < 2)
        {
            System.out.println("args = [file] [N] "
                    +"where file = output filename, N=batch count");
        }
        else
        {
            new GracefulShutdownTest1(
                    new File(args[0]), 
                    Integer.parseInt(args[1])
            ).run();
        }
    }
}

回答:


160

次のことができます。

  • シャットダウンフックで、いくつかのAtomicBoolean(または揮発性ブール)の「keepRunning」をfalseに設定します
  • (オプションで.interrupt、いくつかのブロッキング呼び出しでデータを待つ場合の作業スレッド)
  • 作業スレッドでメソッドをwriteBatch呼び出して、作業スレッド(ケースでは実行中)が終了するのを待ちますThread.join()
  • プログラムを終了します

いくつかの大ざっぱなコード:

  • 追加する static volatile boolean keepRunning = true;
  • 、実行()あなたが変更

    for (int i = 0; i < N && keepRunning; ++i)
        writeBatch(pw, i);
  • )(メインあなたが追加します。

    final Thread mainThread = Thread.currentThread();
    Runtime.getRuntime().addShutdownHook(new Thread() {
        public void run() {
            keepRunning = false;
            mainThread.join();
        }
    });

ターミナルで優雅な「Control-Cを押したときにすべてのクライアントを拒否する」というのは、おおざっぱな方法です。


ドキュメントから:

仮想マシンがシャットダウンシーケンスを開始すると、登録されているすべてのシャットダウンフックが不特定の順序で開始され、同時に実行されます。すべてのフックが終了すると、finalization-on-exitが有効になっている場合は、呼び出されていないすべてのファイナライザが実行されます。最後に、仮想マシンが停止します。

つまり、シャットダウンフックは、フックが終了する(run()メソッドから返されるまで、JVMを実行し続けます。


14
ああ-私が正しく理解していれば、あなたの言っていることの要点は、シャットダウンフックが現在実行中の他のスレッドと通信し、VMシャットダウンを保留する機会があることです(たとえば、Thread.join()を使用することによって) )クイッククリーンアップが完了したことが確認されるまで。それが私が欠けていた点です。ありがとう!
ジェイソンS

2
はい。了解しました。回答をドキュメントのいくつかの文で更新しました。
aioobe

1
事前に申し訳ありませんが、GracefulShutdownTest1はRunnableを実装すべきではありませんか?
ビクター

これも動作しますか?GUIがメインスレッドで開始されましたか?それが最良の方法ではないことはわかっていますが(代わりにEDTを使用)、古いコードで作業しています。
Agostino 2014年

1
@MartynasJusevičiusの場合、デーモン以外のすべてのスレッドが終了すると、JVMが終了します。mainThreadがデーモンスレッドの場合、joinJVMをシャットダウンさせる前に、スレッドが完了するのを待つことが保証されます。
aioobe

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.