前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >springboot项目在docker容器中如何优雅关闭

springboot项目在docker容器中如何优雅关闭

作者头像
lyb-geek
发布2022-01-06 16:48:27
2.9K1
发布2022-01-06 16:48:27
举报
文章被收录于专栏:Linyb极客之路Linyb极客之路

01前言

1什么是优雅关闭

在我看来所谓的优雅关闭,就是在系统关闭时,预留一些时间,让你有机会来善后一些事情

2什么时候需要优雅关闭

是否所有项目都需要优雅关闭?那也不一定,毕竟所谓的优雅关闭,另一面就意味这关闭得慢,因此项目的优雅关闭得看项目的核心程度,换言之就是看该项目处理的数据是不是核心数据,其实项目的最终本质,是对数据的处理。

3如何实现优雅关闭

通用的方法是通过进程接收到系统发送信号变量来实现,什么是信号变量,可以参考如下链接

https://www.cnblogs.com/liuhouhou/p/5400540.html

02在springboot项目中如何配置优雅关闭

1如果是springboot2.3版本之前可以通引入如下jar

代码语言:javascript
复制
<dependency>
    <groupId>com.github.timpeeters</groupId>
    <artifactId>spring-boot-graceful-shutdown</artifactId>
    <version>X.X.X</version>
</dependency>

通过引入这个jar,并且在项目的application.yml文件配置

代码语言:javascript
复制
graceful:
  shutdown:
    enabled: true

只需两步就可实现优雅关闭。当项目进行关闭时,观察日志,会发现有如下输出

spring-boot-graceful-shutdown目前支持的springboot版本如下

其实现流程

更多详细教程可以参考如下链接

https://github.com/timpeeters/spring-boot-graceful-shutdown

2springboot2.3版本之后可以在application.yml做如下配置即可

代码语言:javascript
复制
server:
  # 开启优雅关闭,默认:IMMEDIATE,立即关闭
  shutdown: graceful

spring:
  lifecycle:
    # 配置优雅关闭宽限时间,即项目在30s都没处理完,则进行强制关闭
    timeout-per-shutdown-phase: 30s

其关闭效果如下

代码语言:javascript
复制
2021-01-15 10:52:04.063  INFO 39004 --- [extShutdownHook] o.s.b.w.e.tomcat.GracefulShutdown : Commencing graceful shutdown. Waiting for active requests to complete
2021-01-15 10:52:04.138  INFO 39004 --- [nfoReplicator-0] com.netflix.discovery.DiscoveryClient : DiscoveryClient_HELLO-LOCAL/hello-local:172.17.45.115:8080 - registration status: 204
2021-01-15 10:52:04.605  INFO 39004 --- [tomcat-shutdown] o.s.b.w.e.tomcat.GracefulShutdown : Graceful shutdown complete
c.netflix.discovery.TimedSupervisorTask : task supervisor shutting down, can't accept the task

03在docker容器中如何进行优雅关闭

如果在Dockerfile做如下配置

代码语言:javascript
复制
ENTRYPOINT [ "sh", "-c", "java $JAVA_OPTS -Djava.security.egd=file:/dev/./urandom -jar /app.jar" ]

是没法实现优雅关闭的效果。其原因是使用 docker stop 关闭容器时, 只有 init(pid 1)进程能收到中断信号, 如果容器的pid 1 进程是 sh 进程, 它不具备转发结束信号到它的子进程的能力, 所以我们真正的java程序得不到中断信号, 也就不能实现优雅关闭. 解决思路是: 让pid 1 进程具备转发终止信号, 或者将 java 程序配成 pid 1 进程。因此只需对Dockerfile做如下改造就行

代码语言:javascript
复制
ENTRYPOINT [ "sh", "-c", "exec java $JAVA_OPTS -Djava.security.egd=file:/dev/./urandom -jar /app.jar" ]

其实就是在java前边加上exec即可。其实现机理可以参考如下链接

https://spring.io/guides/topicals/spring-boot-docker

04在k8s中如何进行优雅关闭

1配置preStop Hook钩子

preStop Hook 是一个发送到 Pod 中的容器特殊命令或 Http 请求。如果您的应用程序在接收 SIGTERM 时没有正常关闭,您可以使用 preStop Hook 来触发正常关闭。接收 SIGTERM 时大多数程序都会正常关闭,但如果您使用的是第三方代码或管理的系统无法控制,则 preStop Hook 是在不修改应用程序的情况下触发正常关闭的好方法。

2 适当延长terminationGracePeriodSeconds时间

terminationGracePeriodSeconds这玩意的意思直接看如下

其配置参考如下

代码语言:javascript
复制
apiVersion: apps/v1
kind: Deployment
metadata:
  namespace: dev
  name: hello
  labels:
    app: hello
spec:
  replicas: 3
  selector:
    matchLabels:
      app: hello
  template:
    metadata:
      labels:
        app: hello
    spec:
      imagePullSecrets:
      - name: default-secret
      containers:
      - name: hello
        image: {{images}}
        ports:
          - containerPort: 8080
        lifecycle:
          preStop:
            httpGet:
              port: 8080
              path: 你要关闭时执行的动作
      terminationGracePeriodSeconds: 60

05总结

优雅关闭正常都是会配置一定的处理时间,超过该时间没处理完,就会进行强杀。因此对于核心业务,我们还得考虑万一进行强杀时,还要考虑是否需要对业务进行补偿操作

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2021-01-17,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Linyb极客之路 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
容器服务
腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档