一尘不染

Java中的关闭挂钩的有用示例?

java

我试图确保我的Java应用程序采取合理的步骤来增强功能,并且其中的一部分涉及正常关闭。我正在阅读有关关机挂钩的信息,但实际上并没有实际使用它们的方法。

有没有实际的例子?

假设我有一个非常简单的应用程序,例如下面的程序,它以100为批号将数字写入文件,将10写入行,并且如果程序被中断,我想确保给定的批处理完成。我知道如何注册一个关闭钩子,但是我不知道如何将其集成到我的应用程序中。有什么建议么?

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();
        }
    }
}

阅读 354

收藏
2020-03-09

共1个答案

一尘不染

你可以执行以下操作:

  • 让关闭挂钩将一些AtomicBoolean(或易失性布尔值)“ keepRunning”设置为false
  • (可选地,.interrupt工作线程是否在某些阻塞调用中等待数据)
  • writeBatch通过Thread.join()在工作线程上调用方法,等待工作线程(在你的情况下执行)完成。
  • 终止程序

一些粗略的代码:

  • 添加一个 static volatile boolean keepRunning = true;
  • 在run()中,你更改为
for (int i = 0; i < N && keepRunning; ++i)
    writeBatch(pw, i);
  • 在main()中添加:
final Thread mainThread = Thread.currentThread();
Runtime.getRuntime().addShutdownHook(new Thread() {
    public void run() {
        keepRunning = false;
        mainThread.join();
    }
});

这大概就是我在终端中优雅地“击中Control-C时拒绝所有客户端”的方式。

2020-03-09