偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Java層如何配合K8s實(shí)現(xiàn)優(yōu)雅下線

開發(fā) 前端
在Kubernetes(K8s)中部署Java應(yīng)用(如Spring Boot)時(shí),實(shí)現(xiàn)優(yōu)雅下線(Graceful Shutdown)是確保服務(wù)平滑停止的關(guān)鍵,避免正在處理的請求中斷或數(shù)據(jù)不一致,尤其在金融、支付等高可靠性場景中至關(guān)重要。

在Kubernetes(K8s)中部署Java應(yīng)用(如Spring Boot)時(shí),實(shí)現(xiàn)優(yōu)雅下線(Graceful Shutdown)是確保服務(wù)平滑停止的關(guān)鍵,避免正在處理的請求中斷或數(shù)據(jù)不一致,尤其在金融、支付等高可靠性場景中至關(guān)重要。優(yōu)雅下線需要Java應(yīng)用層與K8s的生命周期管理機(jī)制協(xié)同工作。以下是詳細(xì)實(shí)現(xiàn)方案,涵蓋原理、代碼實(shí)現(xiàn)和配置步驟。

一、優(yōu)雅下線的原理

1. K8s下線流程

  • 當(dāng)K8s執(zhí)行kubectl delete或滾動(dòng)更新(Rolling Update)時(shí):
  1. Pod標(biāo)記為Terminating:K8s向Pod發(fā)送SIGTERM信號。
  2. 負(fù)載均衡移除:Service從Endpoint中移除該P(yáng)od,停止新流量。
  3. 寬限期等待:K8s等待terminationGracePeriodSeconds(默認(rèn)30秒),然后發(fā)送SIGKILL強(qiáng)制終止。
  • 目標(biāo):

在寬限期內(nèi)完成現(xiàn)有請求處理,拒絕新請求,釋放資源。

2. Java層需求

  • 捕獲SIGTERM:監(jiān)聽操作系統(tǒng)信號,觸發(fā)關(guān)閉邏輯。
  • 停止新請求:關(guān)閉Web服務(wù)器(如Tomcat),但保留現(xiàn)有連接。
  • 完成任務(wù):等待異步任務(wù)(如數(shù)據(jù)庫寫入)結(jié)束。
  • 通知K8s:通過健康檢查(Readiness Probe)告知已下線。

二、Java層實(shí)現(xiàn)優(yōu)雅下線

以Spring Boot為例,結(jié)合K8s的preStop鉤子和Spring的關(guān)閉機(jī)制實(shí)現(xiàn)。

1. 捕獲SIGTERM信號

  • Spring Boot通過ApplicationListener監(jiān)聽上下文關(guān)閉事件:
import org.springframework.boot.SpringApplication;
import org.springframework.boot.context.event.ApplicationContextInitializedEvent;
import org.springframework.context.ApplicationListener;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.stereotype.Component;

@Component
public class GracefulShutdownListener implements ApplicationListener<ContextClosedEvent> {
    @Override
    public void onApplicationEvent(ContextClosedEvent event) {
        System.out.println("Received SIGTERM, starting graceful shutdown...");
        // 自定義關(guān)閉邏輯
        try {
            Thread.sleep(5000); // 模擬等待現(xiàn)有請求完成
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }
        System.out.println("Shutdown complete.");
    }
}

2. 關(guān)閉Web服務(wù)器

  • Spring Boot嵌入Tomcat時(shí),需優(yōu)雅停止連接:
import org.apache.catalina.connector.Connector;
import org.springframework.boot.web.embedded.tomcat.TomcatConnectorCustomizer;
import org.springframework.context.ApplicationListener;
import org.springframework.context.event.ContextClosedEvent;
import org.springframework.stereotype.Component;

@Component
public class TomcatGracefulShutdown implements ApplicationListener<ContextClosedEvent>, TomcatConnectorCustomizer {
    private volatile Connector connector;

    @Override
    public void customize(Connector connector) {
        this.connector = connector;
    }

    @Override
    public void onApplicationEvent(ContextClosedEvent event) {
        if (connector != null) {
            System.out.println("Shutting down Tomcat gracefully...");
            connector.pause(); // 暫停新請求
            try {
                Thread.sleep(5000); // 等待現(xiàn)有請求完成,實(shí)際應(yīng)動(dòng)態(tài)判斷
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
            connector.getService().stop(); // 停止服務(wù)
        }
    }
}
  • 配置Spring Boot:
@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        SpringApplication app = new SpringApplication(Application.class);
        app.addListeners(new TomcatGracefulShutdown());
        app.run(args);
    }
}

3. 處理異步任務(wù)

  • 若有線程池或消息隊(duì)列(如Kafka消費(fèi)者),需等待任務(wù)完成:
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationListener;
import org.springframework.context.event.ContextClosedEvent;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;
import org.springframework.stereotype.Component;

@Component
public class AsyncTaskShutdown implements ApplicationListener<ContextClosedEvent> {
    @Autowired
    private ThreadPoolTaskExecutor executor;

    @Override
    public void onApplicationEvent(ContextClosedEvent event) {
        System.out.println("Shutting down async tasks...");
        executor.setWaitForTasksToCompleteOnShutdown(true); // 等待任務(wù)完成
        executor.setAwaitTerminationSeconds(10); // 最多等10秒
        executor.shutdown();
    }
}

三、K8s配置配合

1. 設(shè)置寬限期

  • 在Pod配置中延長terminationGracePeriodSeconds,給Java足夠關(guān)閉時(shí)間:
apiVersion: v1
kind: Pod
metadata:
  name: spring-boot-app
spec:
  containers:
  - name: app
    image: spring-boot-app:latest
    terminationGracePeriodSeconds: 60  # 寬限期60秒

2. 添加preStop鉤子

  • 在容器停止前執(zhí)行腳本,通知應(yīng)用準(zhǔn)備下線:
spec:
  containers:
  - name: app
    image: spring-boot-app:latest
    lifecycle:
      preStop:
        exec:
          command: ["/bin/sh", "-c", "curl -X POST http://localhost:8080/actuator/shutdown"]
    ports:
    - containerPort: 8080
  • Spring Boot啟用Actuator:
management.endpoint.shutdown.enabled=true
management.endpoints.web.exposure.include=shutdown
  • 說明:preStop調(diào)用/actuator/shutdown,觸發(fā)Spring上下文關(guān)閉。

3. 配置Readiness Probe

  • 讓K8s感知應(yīng)用不再就緒,移除流量:
spec:
  containers:
  - name: app
    readinessProbe:
      httpGet:
        path: /actuator/health/readiness
        port: 8080
      initialDelaySeconds: 5
      periodSeconds: 10
  • Java代碼:關(guān)閉時(shí)更新健康狀態(tài):
import org.springframework.boot.actuate.health.Health;
import org.springframework.boot.actuate.health.HealthIndicator;
import org.springframework.stereotype.Component;

@Component
public class ShutdownHealthIndicator implements HealthIndicator {
    private volatile boolean isShuttingDown = false;

    public void setShuttingDown(boolean shuttingDown) {
        this.isShuttingDown = shuttingDown;
    }

    @Override
    public Health health() {
        return isShuttingDown ? Health.down().build() : Health.up().build();
    }
}

@Component
public class ShutdownListener implements ApplicationListener<ContextClosedEvent> {
    @Autowired
    private ShutdownHealthIndicator healthIndicator;

    @Override
    public void onApplicationEvent(ContextClosedEvent event) {
        healthIndicator.setShuttingDown(true); // 標(biāo)記下線
    }
}

四、完整流程

  1. K8s發(fā)起停止:
  • 發(fā)送SIGTERM,觸發(fā)preStop。
  1. Java響應(yīng):
  • /actuator/shutdown關(guān)閉Spring上下文。
  • Readiness Probe返回DOWN,K8s移除流量。
  • Tomcat暫停新請求,等待現(xiàn)有請求完成。
  • 異步任務(wù)執(zhí)行完畢。
  1. Pod終止:
  • 寬限期(60秒)內(nèi)完成,K8s發(fā)送SIGKILL。

五、驗(yàn)證與優(yōu)化

  • 測試:
kubectl delete pod spring-boot-app
# 檢查日志,確保"Shutdown complete"打印,且無請求中斷
  • 優(yōu)化:

動(dòng)態(tài)等待:根據(jù)活躍連接數(shù)(Tomcat getActiveCount)調(diào)整睡眠時(shí)間。

超時(shí)控制:若任務(wù)未完成,記錄日志并強(qiáng)制退出。

六、總結(jié)

  • Java層:通過ContextClosedEvent捕獲信號,優(yōu)雅關(guān)閉Tomcat和異步任務(wù)。
  • K8s配合:設(shè)置terminationGracePeriodSeconds、preStop和Readiness Probe,確保流量移除和資源釋放。
  • 效果:請求零中斷,數(shù)據(jù)一致性保障。
責(zé)任編輯:武曉燕 來源: 科學(xué)隨想錄
相關(guān)推薦

2023-12-26 15:05:00

Linux共享內(nèi)存配置

2022-04-22 13:32:01

K8s容器引擎架構(gòu)

2022-04-29 10:40:38

技術(shù)服務(wù)端K8s

2023-11-06 07:16:22

WasmK8s模塊

2019-04-29 08:41:44

K8S集群節(jié)點(diǎn)

2019-05-17 15:16:24

Kubernetes容器集群

2023-09-06 08:12:04

k8s云原生

2022-06-14 09:49:10

K8sLinuxKubernetes

2023-11-30 08:08:30

k8s優(yōu)雅停機(jī)

2023-08-03 08:36:30

Service服務(wù)架構(gòu)

2023-08-04 08:19:02

2023-05-25 21:38:30

2020-05-12 10:20:39

K8s kubernetes中間件

2022-09-05 08:26:29

Kubernetes標(biāo)簽

2023-09-15 07:34:15

AIOps云原生項(xiàng)目

2021-12-08 07:55:53

K8S WebhookKubernetes

2021-05-07 09:31:33

KindK8s Operator

2023-07-04 07:30:03

容器Pod組件

2022-08-15 09:49:28

K8s云原生

2022-01-11 07:59:15

K8S KubernetesAirflow
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號