Spring系列之HikariCP連線池

程式設計師阿牛發表於2021-08-25

上兩篇文章,我們講到了Spring中如何配置單資料來源和多資料來源,配置資料來源的時候,連線池有很多選擇,在SpringBoot 1.0中使用的是Tomcat的DataSource,在SpringBoot 2.0中,我們使用預設連線池是HikariCP,本文講一下HikariCP。

為什麼SpringBoot 2.0要選擇HikariCP來作為預設的連線池呢?

我們先看一下官網的一張對比圖。
file
一個連線週期定義為單個DataSource.getConnection()/ Connection.close()。 一個語句週期定義為單個Connection.prepareStatement(), Statement.execute(), Statement.close()
從上圖看出,HikariCP和常見的連線池相比,優勢非常明顯。

為什麼HikariCP那麼快呢?根據官網概要總結了以下幾點

  • 位元組碼精簡 :位元組碼級別優化(很多⽅法通過 JavaAssist ⽣成),直到編譯後的位元組碼最少,這樣,CPU快取可以載入更多的程式程式碼;
  • 優化代理和攔截器:減少程式碼,例如HikariCP的Statement proxy只有100行程式碼,只有BoneCP的十分之一;
  • 自定義陣列型別:(FastStatementList)代替ArrayList:避免每次get()呼叫都要進行range check,避免呼叫remove()時的從頭到尾的掃描;
  • 自定義集合型別(ConcurrentBag:提高併發讀寫的效率;
  • 代理類的優化(⽐如,⽤ invokestatic 代替了 invokevirtual)
  • 其他針對BoneCP缺陷的優化,比如對於耗時超過一個CPU時間片的方法呼叫的研究(但沒說具體怎麼優化)。

既然HikariCP那麼快,接下來就看一下在Spring中怎麼使用HikariCP?

在Spring Boot 2.x中
• 預設使⽤ HikariCP
• 配置 spring.datasource.hikari.* 配置

在Spring Boot 1.x中
• 預設使⽤ Tomcat 連線池,需要移除 tomcat-jdbc 依賴
• 在application.properties檔案中加上spring.datasource.type=com.zaxxer.hikari.HikariDataSource

我們來看一下SpringBoot2.0怎麼使用配置HikariDataSource的

下面是org.springframework.boot.autoconfigure.jdbc.DataSourceConfiguration中的相關程式碼


/**
* 下面的三個註解意思是當classpath中有HikariDataSource.class,並且Spring上下文中沒有配置DataSource的bean
* 並且spring.datasource.type的值是com.zaxxer.hikari.HikariDataSource的時候,SpringBoot自動幫我們選擇預設的連線池是HikariDataSource
*/
@ConditionalOnClass({HikariDataSource.class})
@ConditionalOnMissingBean({DataSource.class})
@ConditionalOnProperty(name = {"spring.datasource.type"},havingValue = "com.zaxxer.hikari.HikariDataSource",matchIfMissing = true)
static class Hikari {
        Hikari() {
}

@Bean
@ConfigurationProperties(prefix = "spring.datasource.hikari")
HikariDataSource dataSource(DataSourceProperties properties) {
            HikariDataSource dataSource = (HikariDataSource)DataSourceConfiguration.createDataSource(properties, HikariDataSource.class);
            if (StringUtils.hasText(properties.getName())) {
                dataSource.setPoolName(properties.getName());
            }
            return dataSource;
        }
}

最後看看,HikariCp配置的引數有哪些?

# 不同資料來源這四個配置都會用到
spring.datasource.url=jdbc:mysql://localhost:3306/test
spring.datasource.username=root
spring.datasource.password=123456
spring.datasource.driver-class-name=com.mysql.jdbc.Driver

#以下的配置項是hikari特有的配置
# 等待連線池分配連線的最大時長(毫秒),超過這個時長還沒可用的連線則發生SQLException, 預設:30秒
spring.datasource.hikari.connection-timeout=30000
# 最小連線數
spring.datasource.hikari.minimum-idle=5
# 最大連線數
spring.datasource.hikari.maximum-pool-size=15
# 自動提交
spring.datasource.hikari.auto-commit=true
# 一個連線idle狀態的最大時長(毫秒),超時則被釋放(retired),預設:10分鐘
spring.datasource.hikari.idle-timeout=600000
# 連線池名字
spring.datasource.hikari.pool-name=DatebookHikariCP
# 一個連線的生命時長(毫秒),超時而且沒被使用則被釋放(retired),預設:30分鐘 1800000ms,建議設定比資料庫超時時長少60秒
spring.datasource.hikari.max-lifetime=28740000
spring.datasource.hikari.connection-test-query=SELECT 1

#以下是針對MYSQL驅動的配置引數
# 在每個連線中快取的語句的數量。預設值為保守值25。建議將其設定為250-500之間
spring.datasource.hikari.prepStmtCacheSize = 300
# 快取的已準備SQL語句的最大長度,預設值是256,但是往往這個長度不夠用
spring.datasource.hikari.prepStmtCacheSqlLimit = 2048
# 快取開關,如果這裡設定為false,上面兩個引數都不生效
spring.datasource.hikari.cachePrepStmts = true
#較新版本的 MySQL 支援伺服器端準備好的語句,這可以提供實質性的效能提升
spring.datasource.hikari.useServerPrepStmts = true

HikariCP官方地址: https://github.com/brettwooldridge/HikariCP/wiki

相關文章