工作5年了,你竟然还不会应用优雅停机?

news2024/10/1 12:10:59

事情是这样的,小明是一个工作五年的老程序员,半秃着的头已经彰显了他深不可测的技术实力。

在这里插入图片描述

这一天,小明收到了领导给过来的一个需求。

领导对小明说:“小明啊,你工作五年了,这个需求我交给你一个人负责很是放心,你可要保质保量按时落地呀”
小明当即立下军令状,拍着胸脯说到:“放心领导,交给我了”

在这里插入图片描述

小明承诺之后就开始认真看起了需求,过了五分钟以后,小明一拍脑袋,发现这需求还蛮简单,就是异步发个RocketMQ然后再由下游系统消费去修改Redis的一些信息,于是小明三下五除二,没花几天就把这事给干完了,于是开启了快乐的摸鱼,等待发布的那一天…

在这里插入图片描述

发布上线的这一天在小明的摸鱼下很快到来了…
因为部署的服务器很多,然后需要保证兼容性发布,不能直接停机,所以小明采取的是与往日一贯的兼容性发布方案,也就是分批发布。
准备好所有环境数据以后小明就开始发布了。

在这里插入图片描述

在重启第一批机器时,令人意想不到的事发生了,生产疯狂打印如下报错信息

在这里插入图片描述

吓的小明直接就给回滚了,为此这件事还惊动到了领导,让领导给铺天盖地一顿哐哐哐
接收完领导的批评以后,小明开始认真复盘为什么会出现这样的问题。

在这里插入图片描述

于是小明先画了一个简单的事故回顾分析图来帮助自己分析
正常运行时,应用如下图所示

在这里插入图片描述

小明看着这幅图仔细思索,突然小明敏锐的察觉到,难道是Redis与RocketMQ的关闭顺序导致的吗?带着这个问题点,小明开启了对应的源码解析之路

小明先是分析了Redis关闭核心源码(注意:案例使用的都是标准的SpringBoot集成Redis)

在这里插入图片描述

紧接着小明继续分析了RocketMQ关闭的核心源码(注意:案例使用的都是标准的SpringBoot集成RocketMQ)

在这里插入图片描述

然后小明结合上面两个中间件,继而分析了Spring在进行容器关闭时是如何关闭各类bean的

在这里插入图片描述

果然就是这个问题导致的,我们再来画一个图来梳理一下这个事故

在这里插入图片描述

分析完原因后,小明就开始思考解决方案了,经过几个夜晚的抓耳挠腮以及对Spring源码的阅读,小明终于找到了解决方案

在这里插入图片描述

小明是这样来设计的
步骤一、Spring容器启动快完成时会发送一个ContextRefreshedEvent事件,可以通过这个事件把RocketMQ自己注册的DefaultRocketMQListenerContainer类型的bean取出,存放在一个自己装配的config当中,目的是为了可以自己来控制RocketMQ shutDown的时机
在这里插入图片描述

PS:这里是一定要通过这个事件来做的,不要想着通过@Bean去找或者通过实现ApplicationContextAware接口然后拿到ApplicationContext去getBean的方式获取,因为这个执行过程在RocketMQ自己注册DefaultRocketMQListenerContainer的步骤之前,所以你是拿不到的

拿不到的原因给大家看一下两张源码图
在这里插入图片描述
在这里插入图片描述

步骤二、在应用进行关闭时一般是接收到了kill -9
或者调用了exit方法,这个时候Spring会在关闭时发出closeEvent事件,而这个事件是在destoryBean之前发出来的,这个时候我们就可以提前主动对所有的DefaultRocketMQListenerContainer进行shutDown了

步骤三、Redis的shutDown这个时候就可以放心的交给Spring自己的destoryBean方法来执行了

最后直接给大家看解决方案的代码
在这里插入图片描述

做完这一系列改造之后,于是乎小明再一次信心满满的开始了发布之旅,这一次就很顺利啦,线上没有再出现该类报错问题,顺利的完成了领导布置的任务,经过这件事后小明的实力又变强了几分

在这里插入图片描述

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

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

相关文章

开源写作平台WriteFreely(基础篇)

什么是 WriteFreely ? WriteFreely 是一个专为作家打造的干净、极简主义的出版平台。可以用来创建一个博客,在您的组织内分享知识,或者围绕共同的写作行为建立一个社区。 这里值得一提的是, WriteFreely 支持 ActivityPub 协议,这…

逐句回答,流式返回,ChatGPT采用的Server-sent events后端实时推送协议Python3.10实现,基于Tornado6.1

善于观察的朋友一定会敏锐地发现ChatGPT网页端是逐句给出问题答案的,同样,ChatGPT后台Api接口请求中,如果将Stream参数设置为True后,Api接口也可以实现和ChatGPT网页端一样的流式返回,进而更快地给到前端用户反馈&…

剑指offer在排序数组中的二分法应用总结

排序数组中的搜索问题,首先想到 二分法 解决,本篇详细解析关于二分法边界的问题。 目录 一、二分法概念 二、剑指Offer53.在排序数组中查找数字 三、在排序数组中查找元素的第一个和最后一个位置 一、二分法概念 二分法就是在一个有序递增的数组中进行…

使用RabbitMQ发送短信

1、在项目中分别创建模块financial-core、financial-mq、financial-sms&#xff0c;如图&#xff1a; 模块构成 <modules><module>financial-common</module><module>financial-base</module><module>financial-core</module><mo…

剑指 Offer 66. 构建乘积数组

摘要 剑指 Offer 66. 构建乘积数组 一、左右乘积列表 我们不必将所有数字的乘积除以给定索引处的数字得到相应的答案&#xff0c;而是利用索引左侧所有数字的乘积和右侧所有数字的乘积&#xff08;即前缀与后缀&#xff09;相乘得到答案。对于给定索引i&#xff0c;我们将使…

Qt广告机服务器(上位机)

目录功能结构adSever.promain.cpptcp_MSG.h 共用Tcp传输信息adsever.h 服务器adsever.cpp 服务器addate.h 时间处理addate.cpp 时间处理adtcp.h 客户端Socket处理adtcp.cpp 客户端Socket处理client.h 客户端信息类client.cpp 客户端信息类admsglist.h 信息记录模块admsglist.cp…

jupyter的使用

1.安装 安装过程看这篇记录。 安装 2.如何启动 环境搭建好后&#xff0c;本机输⼊jupyter notebook命令&#xff0c;会⾃动弹出浏览器窗⼝打开 Jupyter Notebook # 进⼊虚拟环境 workon ai(这个是虚拟环境的名称) # 输⼊命令 jupyter notebook本地notebook的默认URL为&…

宝藏级BI数据可视化功能|图表联动分析

在浏览其他人的BI数据可视化报表时&#xff0c;经常会发现这样的一个现象&#xff0c;点一下上一张数据可视化图表中的某个门店&#xff0c;下一张图表将立即针对该门店展开数据可视化分析。这是什么效果&#xff1f;怎么实现&#xff1f;BI软件中还有多少宝藏级BI数据可视化功…

Oracle表分区的创建、新增、拆分

Oracle中为了方便管理、查询数据当数据量大于500w或者2G时最好用分区表&#xff0c;常见的一种是使用时间作为分区。 分区表添加新的分区有 2 种情况&#xff1a; (1) 原分区里边界是 maxvalue 或者 default。 这种情况下&#xff0c;我们需要把边界分区 drop 掉&#xff0c;加…

好的计划是成功的一半,如何制定项目计划?

好的计划是成功的一半&#xff0c;不好的计划会使项目一步步失败&#xff0c;任何事情&#xff0c;要取得成功&#xff0c;离不开一个科学合理的计划。 计划是为了实现项目所提出的各项目标&#xff0c;每一项任务都是针对某一个特定目标的&#xff0c;因此&#xff0c;一项计划…

计算机视觉手指甲标注案例

关键点标注是指识别和标注图像或视频中特定的相关点或区域的过程。在机器学习行业&#xff0c;它经常被用来训练计算机视觉模型&#xff0c;以执行诸如物体检测、分割和跟踪等任务。 关键点注释可用于以下应用&#xff1a; 面部关键点检测&#xff1a;识别图像中人脸上的眼睛…

12.SpringSecurity中OAuth2.0的实现

一、OAuth2.0介绍 1.概念说明 https://oauth.net/2/ 先说OAuth&#xff0c;OAuth是Open Authorization的简写。   OAuth协议为用户资源的授权提供了一个安全的、开放而又简易的标准。与以往的授权方式不同之处是OAuth的授权不会使第三方触及到用户的帐号信息&#xff08;如…

[league/glide]两行代码实现一套强大的图片处理HTTP服务

只要两行代码&#xff0c;就能实现类似对象存储云提供的基于参数的图片处理&#xff0c;比如裁剪、放大、水印、旋转等等。 我们经常使用第三方的对象存储服务&#xff0c;比如七牛云或阿里云&#xff0c;他们都提供了“智能媒体服务”&#xff0c;其实就是在链接上加上各种参…

Vue:(三十四)Vuex及其属性

Vuex的学习更多是代码了&#xff0c;所以就放在一起了&#xff0c;接下来大概说一下吧。概念&#xff1a;专门在Vue中实现集中式状态&#xff08;数据&#xff09;管理的一个Vue插件&#xff0c;对Vue应用中多个组件的共享状态进行集中式的管理&#xff08;读/写&#xff09;&a…

【Git】P4 Git 远程仓库(2)克隆,抓取与拉取

Git 克隆&#xff0c;拉取与抓取git 克隆 clonegit 拉取 fetch、合并 mergegit 抓取 pullgit 克隆 clone 克隆的使用场景很少&#xff0c;举个例子&#xff0c;老板给你一个任务&#xff1a;这个服务的 bug 由你来解决&#xff1a;那么你的第一步就是从云端克隆到本地&#xf…

VMware 搭建 Linux 系统

前言 使用 VMware Workstation 17 Pro 基于CentOS 7.9 镜像搭建 K8S 一主多从本地虚拟服务器环境 主机名IP配置k8s-master192.168.179.214核CPU 8G内存 20G硬盘k8s-node1192.168.179.224核CPU 8G内存 20G硬盘k8s-node2192.168.179.234核CPU 8G内存 20G硬盘VMware 下载安装 VMw…

Spark读取JDBC调优

Spark读取JDBC调优&#xff0c;如何调参一、场景构建二、参数设置1.灵活运用分区列实际问题&#xff1a;工作中需要读取一个存放了三四年历史数据的pg数仓表&#xff08;缺少主键id&#xff09;&#xff0c;需要将数据同步到阿里云 MC中&#xff0c;Spark在使用JDBC读取关系型数…

案例13-localStorage的使用分析

1、背景介绍 大家看下边的逻辑是否能看明白呢&#xff1f; 前端在调用后端接口获取某一个人的评论次数、获赞次数、回复次数。调用之后判断后端返回过来的值。如果返回回来的值是0的话&#xff0c;从缓存中获取对应的值&#xff0c;如果从缓存中获取的评论次数为空那么其他两…

数据结构——线性数据结构(C语言实现单链表详解)

什么是单链表&#xff1f; 单链表就是一种线性的链式数据结构。单链表通过节点来存储线性数据的&#xff0c;单链表不要求连续的物理空间来存储数据。但是&#xff0c;单链表在逻辑结构上是连续的。通常&#xff0c;会有一个头指针指向单链表的首结点因为单链表的结点会存储一…

【云原生】持久化存储之NFS

文章目录介绍一、NFS1. 部署nfs1.1 找一台服务器作为nfs服务端1.2 检查&#xff1a;1.3 创建挂载路径1.4 在nfs服务器启动nfs服务2. 所有node节点部署nfs服务3. 测试—部署nginx应用&#xff0c;使用nfs持久网络存储二、 PV和PVC2.1 PV2.2 PVC2.3 实现流程2.4 PV&PVC挂载步…