「java让程序优雅的退出」java让程序优雅的退出操作
今天给各位分享java让程序优雅的退出的知识,其中也会对java让程序优雅的退出操作进行解释,如果能碰巧解决你现在面临的问题,别忘了关注本站,现在开始吧!
本文目录一览:
java退出程序语句怎么写
System.exit(-1)终止当前正在运行的 Java 虚拟机,退出程序。
其中参数按照惯例,是用非零的参数码表示异常终止。
java中终止程序的执行
方案操作如下:
(1)让程序在一个单独的线程中运行,然后在终止时,可以用线程的终止方法来结束它。
(2)退出Java程序时尽量不要使用java.lang.System的exit方法。Exit 方法可以终止JVM,从而终止程序,但如果同时运行了另一个Java程序,使用exit方法就会让该程序也关闭,这显然不是希望看到的情况!!!!
要退出Java程序,可以使用destory()退出一个独立运行的过程。对于多线程程序,必须要关闭各个非守护线程。
只有在程序非正常退出时,才使用exit方法退出程序
k8s服务优雅退出
部署预估服务,rollingUpdate,使用Jmeter一直发压力请求
发现会有很多失败的请求
确认当前预估服务不支持优雅退出
当前light4j的版本是1.5.14
发现light4j从1.5.15开始支持graceful shutdown
启动预估服务后,直接给java程序发SIGTERM信号,kill -SIGTERM JAVAPID,发现java程序能够优雅退出
但是通过RollingUpdate,发现java程序没有优雅退出
4.0 删除POD
4.1 POD进入Terminating状态
4.2 与此同时,k8s会将POD从对应的service删除
4.3 与此同时,有preStop hook的容器,会先执行preStop hook, 如果preStop hook的时间超出了Grace period,kubelet会发送SIGTERM
4.4 与此同时,没有preStop hook的容器,kubelet发送SIGTERM信号给启动进程
由于预估服务的镜像是通过bash script启动,而不是直接在Dockerfile里面执行java程序,由上面的知识可知,POD删除的时候,kubelet会发送SIGTERM信号给Bash Script进程。
现在会有一个问题,bash是不能传递信号的
所以需要在bash脚本里面捕获信号,并给java程序发SIGTERM信号
通过Bash Trap信号传递给JAVA进程后,发现RollingUpdate操作还是会有失败的请求,压测qps 500的时候,会有20个请求失败,表现为503, connect refuse
分析:
请求持续不断的过来,这个时候进程收到了SIGTERM信号后开始处理没有完成的请求,但是在切流量的过程中,就是kubelet发送SIGTERM信号和把POD从service拿掉的过程中,有一点请求漏过来了
所以需要确保切流量之后,进程收到SIGTERM之前,不要有流量进来。
加上lifecycle preStop
再次测试,RollingUpdate几次,发现没有失败的请求了
如何优雅的关闭java线程
Java中终止线程的方式主要有三种:
1、使用stop()方法,已被弃用。原因是:stop()是立即终止,会导致一些数据被到处理一部分就会被终止,而用户并不知道哪些数据被处理,哪些没有被处理,产生了不完整的“残疾”数据,不符合完整性,所以被废弃。So, forget it!
2、使用volatile标志位
看一个简单的例子:
首先,实现一个Runnable接口,在其中定义volatile标志位,在run()方法中使用标志位控制程序运行:
public class MyRunnable implements Runnable {
//定义退出标志,true会一直执行,false会退出循环
//使用volatile目的是保证可见性,一处修改了标志,处处都要去主存读取新的值,而不是使用缓存
public volatile boolean flag = true;
public void run() {
System.out.println("第" + Thread.currentThread().getName() + "个线程创建");
try {
Thread.sleep(1000L);
} catch (InterruptedException e) {
e.printStackTrace();
}
//退出标志生效位置
while (flag) {
}
System.out.println("第" + Thread.currentThread().getName() + "个线程终止");
}
}
然后,在main()方法中创建线程,在合适的时候,修改标志位,终止运行中的线程。
public class TreadTest {
public static void main(String[] arg) throws InterruptedException {
MyRunnable runnable = new MyRunnable();
//创建3个线程
for (int i = 1; i = 3; i++) {
Thread thread = new Thread(runnable, i + "");
thread.start();
}
//线程休眠
Thread.sleep(2000L);
System.out.println("——————————————————————————");
//修改退出标志,使线程终止
runnable.flag = false;
}
}
最后,运行结果,如下:
第1个线程创建
第2个线程创建
第3个线程创建
--------------------------
第2个线程终止
第1个线程终止
第3个线程终止
3、使用interrupt()中断的方式,注意使用interrupt()方法中断正在运行中的线程只会修改中断状态位,可以通过isInterrupted()判断。如果使用interrupt()方法中断阻塞中的线程,那么就会抛出InterruptedException异常,可以通过catch捕获异常,然后进行处理后终止线程。有些情况,我们不能判断线程的状态,所以使用interrupt()方法时一定要慎重考虑。
答案来源于我的另一个回答:。
java让程序优雅的退出的介绍就聊到这里吧,感谢你花时间阅读本站内容,更多关于java让程序优雅的退出操作、java让程序优雅的退出的信息别忘了在本站进行查找喔。