如何在 Java 中优雅地处理 SIGTERM 信号?
How to process SIGTERM signal gracefully in Java?
假设我们有一个用 java:
编写的如此简单的守护进程
public class Hellow {
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
while(true) {
// 1. do
// 2. some
// 3. important
// 4. job
// 5. sleep
}
}
}
我们使用 start-stop-daemon
对其进行守护进程,默认情况下它会在 --stop
上发送 SIGTERM
(TERM) 信号
假设当前执行的步骤是#2。而此时此刻,我们正在发送 TERM 信号。
执行会立即终止。
我发现我可以使用 addShutdownHook()
处理信号事件,但问题是它仍然会中断当前的执行并将控制权传递给处理程序:
public class Hellow {
private static boolean shutdownFlag = false;
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
// TODO code application logic here
registerShutdownHook();
try {
doProcessing();
} catch (InterruptedException ex) {
System.out.println(ex);
}
}
static private void doProcessing() throws InterruptedException {
int i = 0;
while(shutdownFlag == false) {
i++;
System.out.println("i:" + i);
if(i == 5) {
System.out.println("i is 5");
System.exit(1); // for testing
}
System.out.println("Hello"); // It doesn't print after System.exit(1);
Thread.sleep(1000);
}
}
static public void setShutdownProcess() {
shutdownFlag = true;
}
private static void registerShutdownHook() {
Runtime.getRuntime().addShutdownHook(new Thread() {
public void run() {
System.out.println("Tralala");
Hellow.setShutdownProcess();
}
});
}
}
所以,我的问题是 - 是否可以不中断当前执行,而是在单独的线程(?)中处理 TERM
信号,以便我能够设置 shutdown_flag = True
以便main
中的循环是否有机会优雅地停止?
您可以使用 SignalHandler。请查看此示例:https://dumpz.org/2707213/
它将赶上 INTERRUPTED
信号并取消设置 operating
标志,这反过来将允许应用正常关闭。
我重写了 registerShutdownHook()
方法,现在它可以正常工作了。
private static void registerShutdownHook() {
final Thread mainThread = Thread.currentThread();
Runtime.getRuntime().addShutdownHook(new Thread() {
public void run() {
try {
System.out.println("Tralala");
Hellow.setShutdownProcess();
mainThread.join();
} catch (InterruptedException ex) {
System.out.println(ex);
}
}
});
}
假设我们有一个用 java:
编写的如此简单的守护进程public class Hellow {
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
while(true) {
// 1. do
// 2. some
// 3. important
// 4. job
// 5. sleep
}
}
}
我们使用 start-stop-daemon
对其进行守护进程,默认情况下它会在 --stop
SIGTERM
(TERM) 信号
假设当前执行的步骤是#2。而此时此刻,我们正在发送 TERM 信号。
执行会立即终止。
我发现我可以使用 addShutdownHook()
处理信号事件,但问题是它仍然会中断当前的执行并将控制权传递给处理程序:
public class Hellow {
private static boolean shutdownFlag = false;
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
// TODO code application logic here
registerShutdownHook();
try {
doProcessing();
} catch (InterruptedException ex) {
System.out.println(ex);
}
}
static private void doProcessing() throws InterruptedException {
int i = 0;
while(shutdownFlag == false) {
i++;
System.out.println("i:" + i);
if(i == 5) {
System.out.println("i is 5");
System.exit(1); // for testing
}
System.out.println("Hello"); // It doesn't print after System.exit(1);
Thread.sleep(1000);
}
}
static public void setShutdownProcess() {
shutdownFlag = true;
}
private static void registerShutdownHook() {
Runtime.getRuntime().addShutdownHook(new Thread() {
public void run() {
System.out.println("Tralala");
Hellow.setShutdownProcess();
}
});
}
}
所以,我的问题是 - 是否可以不中断当前执行,而是在单独的线程(?)中处理 TERM
信号,以便我能够设置 shutdown_flag = True
以便main
中的循环是否有机会优雅地停止?
您可以使用 SignalHandler。请查看此示例:https://dumpz.org/2707213/
它将赶上 INTERRUPTED
信号并取消设置 operating
标志,这反过来将允许应用正常关闭。
我重写了 registerShutdownHook()
方法,现在它可以正常工作了。
private static void registerShutdownHook() {
final Thread mainThread = Thread.currentThread();
Runtime.getRuntime().addShutdownHook(new Thread() {
public void run() {
try {
System.out.println("Tralala");
Hellow.setShutdownProcess();
mainThread.join();
} catch (InterruptedException ex) {
System.out.println(ex);
}
}
});
}