Java - 探究Java优雅退出的两种机制

news2025/1/4 20:28:39

文章目录

  • 概述
  • Java优雅停机_ ShutdownHook 机制
    • 步骤
    • Code
  • Java优雅停机_ 信号量机制
    • SignalHandler 工作原理
    • 使用步骤
    • Linux支持的信号量
    • 根据操作系统选择信号量
    • Code
  • 注意事项

在这里插入图片描述

在这里插入图片描述


概述

在Linux上通过kill -9 pid方式强制终止进程的副作用,这种方式虽然简单高效,但也会带来一些问题,特别是对于应用软件而言。这些问题包括但不限于:

  1. 数据丢失:由于缓存中的数据尚未持久化到磁盘中,强制终止进程可能导致数据丢失,尤其是对于数据库等需要持久化数据的应用。

  2. 文件损坏:如果进程正在进行文件写操作,但尚未完成,突然退出可能导致文件损坏,进而影响数据完整性。

  3. 请求丢失:如果线程的消息队列中有未处理的请求消息,强制终止进程可能导致请求消息丢失,进而影响系统的正常运行。

  4. 应答消息未返回:如果数据库操作已经完成,但应答消息尚未返回给客户端,强制终止进程可能导致客户端等待超时,带来重复更新等问题。

  5. 句柄资源未释放:强制终止进程可能导致句柄资源没有及时释放,从而影响系统的性能和稳定性。

综上所述,虽然强制终止进程是一种简单高效的方式,但在实际应用中需要谨慎使用,尤其是对于需要保证数据完整性和系统稳定性的应用场景,建议使用更加安全可靠的方式来终止进程,比如通过正常的关闭流程来释放资源和保证数据一致性。


Java优雅停机_ ShutdownHook 机制

Java的优雅停机通常通过注册JDK的ShutdownHook来实现,当系统接收到退出指令时,首先标记系统处于退出状态,不再接收新的消息,然后将积压的消息处理完,最后调用资源回收接口将资源销毁,各线程退出执行。

Java的ShutdownHook(关闭钩子)是一种机制,允许开发人员在Java虚拟机(JVM)即将关闭时执行一些特定的代码。

这些代码通常用于释放资源、保存状态或执行清理操作,以确保应用程序在退出时能够完成一些必要的步骤。

ShutdownHook提供了一种优雅退出的机制,使得应用程序可以在正常关闭时执行一些清理工作,而不会因为突然的中断而丢失数据或状态。

步骤

  1. 注册ShutdownHook: 在Java中,可以通过Runtime类的addShutdownHook(Thread hook)方法来注册ShutdownHook。注册的ShutdownHook是一个线程对象,当JVM即将关闭时,会依次执行这些线程对象的代码。

  2. 执行时机: ShutdownHook在以下情况下会被执行:

    • 程序正常退出(调用System.exit(int status)方法)
    • 用户按下Ctrl+C终止程序
    • 操作系统关闭
    • JVM崩溃
  3. 执行顺序: 注册的ShutdownHook会按照注册的顺序依次执行。但是不能保证所有的ShutdownHook都会被执行,因为在某些情况下,比如JVM崩溃,kill -9可能无法正常执行ShutdownHook

  4. 注意事项:

    • ShutdownHook应该尽量保持简单和快速执行,不应该执行过多的操作或者占用太多的时间。
    • 不建议在ShutdownHook中执行一些需要等待的操作,比如等待网络连接、等待I/O操作完成等,因为在JVM关闭时时间是有限的,不能保证这些操作能够正常完成。

Code

package com.artisan.nettycase.a01exist;

/**
 * @author 小工匠
 * @version 1.0
 * @mark: show me the code , change the world
 */
public class ShutdownHookExample {

    public static void main(String[] args) {
        Runtime.getRuntime().addShutdownHook(new Thread(() -> {
            System.out.println("ShutdownHook executed.");
            // 在这里执行清理操作或者释放资源的代码
        }));

        // 程序正常执行
        System.out.println("Program is running...");

        // 模拟程序执行
        try {
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        System.exit(-1);
    }
}
    

当程序执行到System.exit(int status)或者不写System.exit(int status)手工按下Ctrl+C终止程序时,注册的ShutdownHook会被执行,输出"ShutdownHook executed."的信息。

在这里插入图片描述


Java优雅停机_ 信号量机制

除了注册ShutdownHook,还可以通过监听信号量并注册SignalHandler 的方式实现优雅退出。

这种方式通常用于在Unix/Linux系统上,通过捕获特定的信号来执行一些清理操作。Java提供了sun.misc.Signalsun.misc.SignalHandler类来支持这种方式。

SignalHandler 工作原理

SignalHandler的工作原理是通过Java的本地方法接口(JNI)与底层操作系统交互来实现的。具体来说,SignalHandler在Java中是一个接口,它由sun.misc包下的Java类提供。当注册了SignalHandler之后,Java程序会通过JNI调用底层操作系统提供的信号处理函数,将Java程序的信号处理器与操作系统的信号处理机制关联起来。

具体步骤如下:

  1. 注册SignalHandler: Java程序调用sun.misc.Signal.handle(Signal signal, SignalHandler handler)方法注册SignalHandler,将信号对象和信号处理器对象关联起来。

  2. JNI调用: JVM通过JNI调用底层操作系统提供的信号处理函数,将Java程序中注册的SignalHandler传递给操作系统。

  3. 信号处理函数: 底层操作系统收到相应的信号(比如Ctrl+C中断信号),会调用注册的信号处理函数。

  4. 调用Java程序中的信号处理器: 信号处理函数在收到信号后会调用Java程序中注册的信号处理器,执行相应的处理逻辑。

  5. 执行清理操作: Java程序中的信号处理器执行相应的清理操作或释放资源的代码。

总的来说,SignalHandler的工作原理是通过JNI接口与底层操作系统交互,将Java程序中的信号处理器与操作系统的信号处理机制连接起来,实现了对特定信号的监听和处理。


使用步骤

通过监听信号量并注册SignalHandler的方式实现优雅退出的步骤如下:

  1. 创建SignalHandler对象: 首先,需要创建一个实现了sun.misc.SignalHandler接口的信号处理器对象。这个对象将负责处理接收到的信号。

  2. 实现handle方法: 在SignalHandler对象中实现handle(Signal signal)方法,该方法定义了接收到信号时需要执行的操作。在这个方法中,你可以执行一些清理操作、释放资源等。

  3. 创建Signal对象: 使用sun.misc.Signal类创建信号对象,指定要监听的信号名称。

  4. 注册SignalHandler: 调用sun.misc.Signalhandle(Signal signal, SignalHandler handler)方法注册信号处理器,将信号对象和信号处理器对象关联起来。

  5. 监听信号: 通过监听信号对象,等待接收信号。


Linux支持的信号量

信号名称作用
SIGKILL终止进程,强制杀死进程
SIGTERM终止进程,软件终止信号
SIGTSTP停止进程,终端来的停止信号
SIGUSR1终止进程,用户定义信号1
SIGUSR2终止进程,用户定义信号2
SIGINT终止进程,中断进程
SIGQUIT建立core文件终止进程,并且生成core文件

根据操作系统选择信号量

String signalName = System.getProperties().getProperty("os.name")
        .toLowerCase().startsWith("win") ? "INT" : "TERM";

// 使用signalName来处理信号
if (signalName.equals("INT")) {
    // Windows操作系统,选择SIGINT信号
    // 执行SIGINT信号的处理逻辑
} else {
    // 非Windows操作系统,选择SIGTERM信号
    // 执行SIGTERM信号的处理逻辑
}

根据这段代码,如果是Windows操作系统,则选择SIGINT信号,用于接收Ctrl+C中断的指令;如果不是Windows操作系统,则选择SIGTERM信号,用于接收kill pid指令。



Code

演示如何通过监听信号量并注册SignalHandler实现优雅退出 。

 package com.artisan.nettycase.a01exist;

import sun.misc.Signal;
import sun.misc.SignalHandler;

/**
 * @author 小工匠
 * @version 1.0
 * @mark: show me the code , change the world
 */

public class SignalHandlerExample {
    public static void main(String[] args) {
        SignalHandler handler = new SignalHandler() {
            @Override
            public void handle(Signal signal) {
                System.out.println("Received signal: " + signal.getName());
                // 在这里执行清理操作或者释放资源的代码
                // 正常退出程序
                System.exit(0); 
            }
        };

        String sig = System.getProperties().getProperty("os.name").toLowerCase().startsWith("win") ? "INT" : "TERM";

        // 注册SIGINT信号处理器
        Signal.handle(new Signal(sig), handler);


        // 模拟程序执行
        while (true) {
            // 在这里执行业务逻辑
            try {
                Thread.sleep(1000); // 模拟程序执行
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

在上述示例中,创建了一个SignalHandler对象来处理SIGINT或者SIGTERM信号,当接收到这些信号时,会执行handle(Signal signal)方法中的清理操作,并正常退出程序。通过调用Signal.handle(Signal signal, SignalHandler handler)方法来注册信号处理器。

在这里插入图片描述


注意事项

使用Java的ShutdownHook(注册在JVM层面的钩子)进行应用的优雅退出时,有几个重要的注意事项:

  1. ShutdownHook的执行不确定性:ShutdownHook并非在所有情况下都会被JVM执行。例如,如果JVM因为某种错误而崩溃,或者接收到某些信号量(如SIGKILL),又或者尝试去杀掉一个不存在的进程(kill -9 pid),ShutdownHook可能不会被执行。因此,不应依赖ShutdownHook来确保某些资源的绝对释放,特别是对于那些对系统稳定性至关重要的资源。

  2. ShutdownHook的执行顺序:JVM不保证ShutdownHook的执行顺序,如果在一个应用中注册了多个ShutdownHook,它们可能不会按照添加的顺序执行,因此不应该在ShutdownHook中依赖于这个顺序来处理资源。

  3. 动态添加或移除ShutdownHook的限制:在JVM关闭期间,不能再动态地添加或移除ShutdownHook。因此,所有的Hook必须在JVM启动时就已经设置好。

  4. 避免在ShutdownHook中调用System.exit():如果在ShutdownHook中调用System.exit(),会导致当前的JVM进程卡住,无法正常退出。这是因为System.exit()会触发终结器(Terminator)进程,这是一个操作系统级别的操作,它会等待当前进程中的所有线程都结束之后,才会释放资源并退出进程。因此,如果在ShutdownHook中调用System.exit(),会导致资源无法正确释放,从而可能引发资源泄漏。

  5. 对于采用注册SignalHandler实现优雅退出的程序,在handle接口中一定要避免阻塞操作,否则它会导致已经注册的ShutdownHook无法执行,系统也无法退出 。

综上所述,ShutdownHook是一个很有用的特性,但是使用时需要谨慎,主要是为了确保资源的正确释放和应用程序的优雅退出。不过,对于那些特别关键的资源,最好还是有其他更可靠的机制来确保它们在JVM终止之前被正确释放。

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1508867.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

git代码回退

对于代码的回退主要有 git reset 、git revert 1.git reset git reset commitId --soft:回退当前代码仓库到指定提交commitId,当前HEAD和commitId之间的修改会保留,这些修改会在暂存区。就是保留了add的状态git reset commitId --hard&…

WPF —— TextBlock、LineBreak RadioButton控件详解

一:TextBlock 1&#xff1a;TextBlock 简介 <LineBreak/> 换行 显示文本 标签内容和content属性共存 2、TextBlock 常用的属性 Foreground&#xff1a;TextBlock的文本内容的颜色。 Background&#xff1a;背景&#xff0c;获取或设置要用于填充内容区域背景的 Brush…

2-LINUX--Linux 系统文件类型与文件权限

一.文件类型 Linux 下所有的东西都可以看做文件&#xff0c;Linux 将文件分为以下几种类型&#xff1a; 1. 普通文件 ‘-’ 2. 目录文件 ‘d’ 3. 管道文件 ‘p’ 4. 链接文件 ‘l’ 5. 设备文件&#xff08;块设备 ’b’ 、字符设备 ‘c’&#xff09; 6. 套接字…

二叉搜索树题目:前序遍历构造二叉搜索树

文章目录 题目标题和出处难度题目描述要求示例数据范围 解法一思路和算法代码复杂度分析 解法二思路和算法代码复杂度分析 解法三思路和算法代码复杂度分析 解法四思路和算法代码复杂度分析 题目 标题和出处 标题&#xff1a;前序遍历构造二叉搜索树 出处&#xff1a;1008. …

2024年春招程序员个人简历范本(精选5篇|附模板)

HR浏览一份简历也就25秒左右,如果你连「好简历」都没有,怎么能找到好工作呢? 如果你不懂得如何在简历上展示自己,或者觉得怎么改简历都不出彩,那请你一定仔细读完。 Java开发工程师简历范本> 性别 男 年龄 24 学历 本科 张三 专业 计算机科学与技术 毕业院校 …

Python3虚拟环境之pipenv

pipenv是python官方推荐的包管理工具&#xff0c;集成了virtualenv, pip和pyenv三者的功能。集合了所有的包管理工具的长处&#xff0c;自动为项目创建和管理虚拟环境。 安装 pip install pipenv在Pycharm中使用 修改Pipfile的安装源参数url&#xff0c;改为https://pypi.tun…

20240309web前端_第一周作业_古诗词

作业三&#xff1a;古诗词 成果展示&#xff1a; 完整代码&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0&q…

BUUCTF---easyre1

1.记录一下第一次做逆向题目 2.题目描述&#xff0c;下载附件 3.解压之后是一个可执行文件&#xff0c;先用PE查看是否有壳 4.没有壳&#xff0c;接下来用ida打开&#xff0c;直接拖进ida即可&#xff0c;接下来使用快捷键fnshiftf12查看字符&#xff0c;若是没有出现搜索框&a…

探究精酿啤酒的秘密:原料中的天然酵母与纯净水质

在啤酒的世界中&#xff0c;Fendi Club精酿啤酒以其与众不同的口感和深远的余味吸引了全球的啤酒爱好者。而这一切&#xff0c;都归功于其选用的上好原料&#xff0c;特别是天然酵母和纯净水质。 天然酵母是啤酒的灵魂。与工业生产的啤酒酵母不同&#xff0c;天然酵母富含丰富的…

navicat过期了,直接用idea连接mysql

1、我的是社区版&#xff0c;需要下载一个插件&#xff0c;直接搜索安装即可。 2、找到data source&#xff0c;点击mysql 3、你们熟悉的&#xff0c;输入账户密码&#xff0c;点击test Connection测试是否连接成功 4、这个本来是在右边&#xff0c;但是你可以把他挪到左边。 5…

Neo4j 批量导入数据 从官方文档学习LOAD CSV 命令 小白可食用版

学习LOAD CSV&#x1f680; 在使用Neo4j进行大量数据导入的时候&#xff0c;发现如果用代码自动一行一行的导入效率过低&#xff0c;因此明白了为什么需要用到批量导入功能&#xff0c;在Neo4j中允许批量导入CSV文件格式&#xff0c;刚开始从网上的中看了各种半残的博客或者视频…

Servlet容器部署教程

Servlet容器介绍 Sercvlet是基于java的动态网站开发技术&#xff0c;其所有类和组件都是基于java实现的&#xff0c;要想使用Servlet&#xff0c;就必须提前配置好java运行环境。Servlet基于java&#xff0c;可以使用几乎全部的java API&#xff0c;所以它的功能异常强大&…

Oracle 层级查询(Hierarchical Queries)

如果一张表中的数据存在分级&#xff08;即数据间存在父子关系&#xff09;&#xff0c;利用普通SQL语句显示数据间的层级关系非常复杂&#xff0c;可能需要多次连接才能完整的展示出完成的层级关系&#xff0c;更困难的是你可能不知道数据到底有多少层。而利用Oracle的层级查询…

Android Framework 通过脚本动态修改应用私有文件执行权限

你只活一次 要悦己 脚本配置 Android_source/device/sprd/***/test/test_chmod.rc service test_chmod /vendor/bin/test_chmod.shuser rootdisabledoneshoton property:sys.test_chmodtruestart test_chmodAndroid_source/device/sprd/***/test/test_chmod.sh #!/system/bin/…

【CRC】一文搞懂CRC-8 SAE J1850 ZERO校验和

CRC在线计算 一、什么是 CRC 校验和 CRC —— Cyclic redundancy check 循环冗余校验&#xff0c;一种校验接收到的数据是否完整的算法&#xff0c;广泛应用于数据通信&#xff0c;大概流程如下 二、CRC-8 如何计算 首先&#xff0c;想要确定一个 CRC 算法&#xff0c;我们需…

论文笔记:Evaluating the Performance of Large Language Models on GAOKAO Benchmark

1 论文思路 采用zero-shot prompting的方式&#xff0c;将试题转化为ChatGPT的输入 对于数学题&#xff0c;将公式转化为latex输入 主观题由专业教师打分 2 数据 2010~2022年&#xff0c;一共13年间的全国A卷和全国B卷 3 结论 3.1 不同模型的zeroshot 高考总分 3.2 各科主…

web自动化测试框架都是有哪些?

Web自动化测试框架主要有以下几种&#xff1a; 1.Selenium&#xff1a;轻量级的Web自动化测试框架&#xff0c;支持多种Web浏览器和语言的集成。Selenium提供了一个IDE来录制和运行自动化测试脚本&#xff0c;还提供了WebDriver&#xff0c;可以通过编程语言编写自动化测试脚本…

知识积累(四):无

文章目录 1. KL散度2. GELU 激活函数3. 向量运算4. bert4.1 词嵌入4.2 cross-encoder 模型4.3 bert 架构4.4 bert 池化操作 5. Fid 模型&#xff08;Fusion-in-Decoder&#xff09;6. 多分类损失函数6.1 交叉熵损失6.2 softmax 损失 7. t-sne8. NDCG参考资料 1. KL散度 衡量两…

Skia最新版CMake编译

运行示例&#xff1a;example/HelloWorld.cpp Skia: 2024年03月08日 master分支: 993a88a663c817fce23d47394b574e19d9991f2f 使用CMake编译 python tools/git-sync-depsbin/gn gen out/config --idejson --json-ide-script../../gn/gn_to_cmake.py此时output目录会生成CM…

动态规划刷题总结(入门)

目录 什么是动态规划算法 如何判断题目中将使用动态规划算法&#xff1f; 动态规划题目做题步骤 动态规划题目解析 泰波那契数模型 第 N 个泰波那契数 三步问题 使用最小花费爬楼梯 路径问题 不同路径 不同路径 Ⅱ 珠宝的最高价值 下降最短路径和 地下城游…