自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

系統(tǒng)設(shè)計 | Java 應(yīng)用中的配置含義和避坑

開發(fā) 前端
關(guān)于 Mybatis 的 map-underscore-to-camel-case 配置有一個坑,這個配置的含義是把數(shù)據(jù)庫列名中的下劃線自動映射為 Java 對象中的駝峰命名。

這篇文章整理了 Spring Boot 和常見的應(yīng)用中間件配置含義,了解這些配置的目的和原理,避免因為錯誤配置導(dǎo)致生產(chǎn)出現(xiàn)問題,特別是有一些安全問題。

PS:寫下來發(fā)現(xiàn)東西非常多,很多時候我們都只是拷貝過來改改沒問題就不管了,但是這樣囫圇吞棗,會給項目帶來風(fēng)險。

1. Spring Boot 相關(guān)

優(yōu)雅停機

優(yōu)雅停機是指當應(yīng)用接收到停機信號時,能夠妥善地處理正在進行的請求,釋放資源,并在完成這些工作后再停止應(yīng)用。

如果不開啟優(yōu)雅停機,有可能在部署的過程中讓少量未完成的任務(wù)和請求直接終止,帶來意想不到的問題。

默認情況下,Spring Boot 沒有啟用優(yōu)雅停機,而且往往需要和云環(huán)境配合使用。

在 Spring Boot 中的配置方式為(本文以 yaml 的格式):

server:
  shutdown: graceful

同時可以設(shè)置一個優(yōu)雅停機的超時時間,如果在超時時間內(nèi)請求沒有完成,應(yīng)用將強制停機。

spring:
  lifecycle:
    timeout-per-shutdown-phase: 30s

Kubernetes 在停止 Pod 時,會先發(fā)送一個 SIGTERM,并通過 Readiness Probe和Liveness Probe 兩個探針來決定是否釋放容器資源。

探針就是應(yīng)用通過一個 API(可以是 HTTP 或者 TCP,通常都是 HTTP)告訴 Kubernetes 它當前的狀態(tài),讓 Kubernetes 來決策何時重啟,關(guān)于優(yōu)雅停機的內(nèi)容比較多,后面單獨一篇文章討論。

在 Spring Boot 中,探針就是 Spring Boot 的 health 接口,可以通過 Indicator 配置。

Indicator

Spring Boot 提供了一些健康狀態(tài)的 API,這樣就可以給云平臺優(yōu)雅停機使用,也可以提供給監(jiān)控系統(tǒng)用來撥測,如果系統(tǒng)長時間不健康,可以進行告警。

在代碼中實現(xiàn)健康狀態(tài)的類叫做 Indicator,基本上默認配置的 Indicator 就夠用了,但有時候需要根據(jù)自己需要配置一些 Indicator。

比如依賴了一個重要的三方系統(tǒng),這個三方系統(tǒng)不啟動起來,當前系統(tǒng)啟動了也沒意義,于是就可以加一個 Indicator,甚至把三方系統(tǒng)的狀態(tài)暴露到當前系統(tǒng)的健康狀態(tài)信息中。

暴露相關(guān)健康 API 需要引入一個 actuator 依賴:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-actuator</artifactId>
</dependency>

下面是一個例子:

import org.springframework.boot.actuate.health.Health;
import org.springframework.boot.actuate.health.HealthIndicator;
import org.springframework.stereotype.Component;

@Component
public class CustomHealthIndicator implements HealthIndicator {

    @Override
    public Health health() {
        boolean isHealthy = checkSomeServiceHealth();
        
        if (isHealthy) {
            return Health.up().withDetail("customService", "UP").build();
        } else {
            return Health.down().withDetail("customService", "DOWN").build();
        }
    }

    private boolean checkSomeServiceHealth() {
        // 檢查邏輯
        return true; 
    }
}

訪問 /actuator/health 接口,返回結(jié)果大概像下面這樣:

{
  "status": "UP",
  "components": {
    "db": {
      "status": "UP",
      "details": {
        "database": "MySQL",
        "validationQuery": "isValid()"
      }
    },
    "customService": {
      "status": "UP",
      "details": {
        "CustomService": "UP"
      }
    },
    ……
  }
}

打開相關(guān)配置:

management:
  endpoints:
    web:
      exposure:
        include: health,info

在這個配置中,info 類似 health, 提供了一些服務(wù)信息,例如名稱、版本之類的,但是要注意避免把敏感信息從這個接口中暴露出去了。

Actuator

提到了 Actuator,這里有一些配置是不能在生產(chǎn)環(huán)境開啟的,這是比較常見的錯誤,需要注意。

Actuator 除了提供了 health,info 兩個接口,還提供了一堆接口,方便觀察 Spring Boot 應(yīng)用,這些接口都可以在開發(fā)環(huán)境開啟。例如:

  • /health: 顯示應(yīng)用的健康狀態(tài)及詳細信息。
  • /info: 顯示應(yīng)用的一些基本信息(例如版本、描述等)。
  • /env: 查看和調(diào)試環(huán)境屬性,了解配置項的來源。
  • /beans: 用于調(diào)試 Bean 的創(chuàng)建和依賴關(guān)系。
  • /metrics: 查看應(yīng)用的性能指標,如內(nèi)存使用情況、GC 活動等。
  • /httptrace: 用于查看最近 HTTP 請求的詳細信息。
  • /mappings: 用于查看所有請求映射的路徑,方便調(diào)試路由問題。

這些接口開啟后會造成安全、性能問題。

所以推薦的配置如下。

開發(fā)環(huán)境:

management:
  endpoints:
    web:
      exposure:
        include: "*"
  endpoint:
    health:
      show-details: always  # 顯示詳細健康信息,方便調(diào)試

endpoints 只是暴露外部是否可以訪問,實際的功能需要單獨開啟,health,info,metrics 三個接口是默認開啟的。

如果需要打開 beans,可以單獨開啟:

management:
  logfile:
    enabled: true           # 允許查看日志文件,方便調(diào)試
  env:
    enabled: true           # 允許查看環(huán)境變量配置
  configprops:
    enabled: true           # 允許查看配置屬性,幫助調(diào)試
  beans:
    enabled: true           # 允許查看 Bean 信息,調(diào)試依賴關(guān)系
  heapdump:
    enabled: true           # 啟用 Heap Dump,用于內(nèi)存分析
  threaddump:
    enabled: true           # 啟用線程轉(zhuǎn)儲,用于線程分析
  mappings:
    enabled: true           # 允許查看所有請求映射,調(diào)試路由問題
  httptrace:
    enabled: true           # 啟用 HTTP 請求追蹤

而生產(chǎn)環(huán)境,需要將其關(guān)閉,只保留需要開啟的配置:

management:
  endpoints:
    web:
      exposure:
        include: "health,info,metrics"
  endpoint:
    health:
      show-details: never   # 隱藏健康檢查的詳細信息,防止敏感數(shù)據(jù)泄露

日志

日志配置錯誤會導(dǎo)致磁盤被日志寫滿,另外日志級別過低,性能會急劇下降。

在以前還不是容器時代,我們常常使用日志文件存儲日志,再使用一些工具轉(zhuǎn)存走,有時候清理日志的腳本失效,導(dǎo)致磁盤被日志寫爆的場景非常多。

下面是一個在容器環(huán)境下 Spring Boot 默認日志庫的配置:

logging:
  level:
    root: INFO
    org.springframework: WARN #這里放上特定包的日志配置
  pattern:
    console: "%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n"
  file:
    enabled: false # 生產(chǎn)環(huán)境通常不直接寫入文件,而是由 K8s 日志收集系統(tǒng)處理
  stdout:
    enabled: true

在生產(chǎn)上我們一般將日志級別設(shè)置為 INFO,并關(guān)閉文件輸出,而是將日志輸出到 stdout 中,由容器捕獲。

在開發(fā)環(huán)境,我們通常把日志設(shè)置為 DEBUG,更加方便調(diào)試。

秘鑰和口令

正常情況下,大多數(shù)應(yīng)用都不會把口令存放到配置文件中,敏感信息需要放到秘鑰管理系統(tǒng)中(Key Management System)。

例如,在 k8s 中,我們可以使用 Secrets 代替明文的 ConfigMap;云平臺往往提供了相關(guān)的 KMS 產(chǎn)品,例如 Alicloud KMS。

2.Mysql

這里給出一個 Mysql 和 Mybatis 的典型配置,并解釋一下關(guān)鍵配置的含義和避坑經(jīng)驗。

spring:
  datasource:
    url: jdbc:mysql://localhost:3306/mydatabase?useUnicode=true&characterEncoding=utf-8&useSSL=false&serverTimezone=UTC&autoReconnect=true&rewriteBatchedStatements=true
    username: your_username
    password: your_password
    driver-class-name: com.mysql.cj.jdbc.Driver
    hikari:
      maximum-pool-size: 10
      minimum-idle: 5
      idle-timeout: 30000
      max-lifetime: 1800000
      connection-timeout: 30000

mybatis:
  mapper-locations: classpath*:mapper/*.xml
  type-aliases-package: com.example.project.domain
  configuration:
    map-underscore-to-camel-case: true
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl

連接字符串中的配置:

  • useUnicode:啟用了 Unicode 支持。
  • characterEncoding:字符編碼,最好在 useUnicode 配置后,characterEncoding 也明確配置一下,否則它會選擇一個默認的 Unicode 字符編碼。另外,我們現(xiàn)在用的更多的是 utf8mb4,它是 UTF8 的拓展字符集,可以處理更多特性,例如表情,最多占用 4 個字節(jié)的空間。
  • useSSL:如果啟用,需要證書,一般我們都沒有配置。
  • serverTimezone:關(guān)于時區(qū),我們一般都不讓數(shù)據(jù)庫決策,而讓應(yīng)用決定寫入數(shù)據(jù)的時區(qū),這個問題我在以前的文章,《系統(tǒng)設(shè)計中需要考慮到的時間問題》,做過詳細說明。
  • autoReconnect:自動重連,如果不開啟的話,數(shù)據(jù)庫重啟了應(yīng)用也必須重啟。
  • rewriteBatchedStatements:自動優(yōu)化批量插入時的性能。

關(guān)于 driver-class-name,對于 MySQL Connector/J 8.0 以上,類名換成了 com.mysql.cj.jdbc.Driver,舊版本是 com.mysql.jdbc.Driver。

關(guān)于 hikari 配置的含義:

  • maximum-pool-size: 最大連接數(shù),連接池中的最大連接數(shù)。
  • minimum-idle: 最小空閑連接數(shù)。
  • idle-timeout: 空閑連接被回收前的最大等待時間(毫秒)。
  • max-lifetime: 連接在池中的最長存活時間(毫秒)。
  • connection-timeout: 獲取連接的超時時間(毫秒)。

hikari 的配置只是建議值,hikari 配置邏輯是什么呢?一般是基于性能測試反復(fù)調(diào)整,但還是有一些規(guī)律。

這里有個坑,有時候為了優(yōu)化性能,提高了最大連接數(shù)。但一般數(shù)據(jù)庫的連接數(shù)是有限制的,比如 1000。假設(shè)一個系統(tǒng)共同一個Mysql實例,系統(tǒng)共有 10 個服務(wù),每個服務(wù)如果有 10 個容器,最大連接數(shù)最多就只能配置到 10 了,否則就會報沒有鏈接的錯誤(而且是偶爾出現(xiàn)這類問題)。

maximumPoolSize 通常設(shè)置為數(shù)據(jù)庫的并發(fā)連接限制的 50% 到 80% 之間,單個容器允許 10 個 Mysql 連接并不大,maximum-pool-size 可以在 10 - 50 之間調(diào)整。

connection-timeout 過短,在數(shù)據(jù)庫負載高或網(wǎng)絡(luò)不穩(wěn)定的情況下,可能導(dǎo)致頻繁的連接超時,可以嘗試往長一點調(diào)整。

max-lifetime、minimum-idle 取決于負載情況,如果持續(xù)負載比較高,可以設(shè)置長一些,不用為數(shù)據(jù)庫節(jié)省資源,讓連接長時間保持。

關(guān)于 Mybatis 的 map-underscore-to-camel-case 配置有一個坑,這個配置的含義是把數(shù)據(jù)庫列名中的下劃線自動映射為 Java 對象中的駝峰命名。例如,user_name 列將映射為 userName 屬性。但有的時候,命名不規(guī)范,有些詞可能是一個詞組而沒有大寫,會導(dǎo)致匹配失敗。

責(zé)任編輯:武曉燕 來源: TechLead 少個分號
相關(guān)推薦

2020-08-26 07:37:25

Nacos微服務(wù)SpringBoot

2010-07-13 15:31:49

ftp命令

2024-03-28 12:51:00

Spring異步多線程

2018-01-20 20:46:33

2024-04-24 13:45:00

2024-04-03 12:30:00

C++開發(fā)

2021-02-26 00:46:11

CIO數(shù)據(jù)決策數(shù)字化轉(zhuǎn)型

2020-06-12 11:03:22

Python開發(fā)工具

2022-07-12 08:24:01

隊列容量調(diào)度hadoop

2020-12-16 10:00:59

Serverless數(shù)字化云原生

2022-09-26 09:53:18

開發(fā)緩存

2023-04-12 08:18:40

ChatGLM避坑微調(diào)模型

2018-09-05 11:51:55

數(shù)據(jù)庫NoSQLMongoDB

2021-02-22 17:00:31

Service Mes微服務(wù)開發(fā)

2021-05-08 12:30:03

Pythonexe代碼

2023-05-24 10:06:42

多云實踐避坑

2021-05-07 21:53:44

Python 程序pyinstaller

2022-03-04 18:11:16

信服云

2023-10-11 11:27:46

調(diào)試版本程序

2023-05-23 07:51:57

硬盤顆粒SSD
點贊
收藏

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