Tomcat 的 JDBC 連接池
Tomcat 的 JDBC 連接池簡介
JDBC 連接池 org.apache.tomcat.jdbc.pool 是 Apache Commons DBCP 連接池的一種替換或備選方案。
那究竟為何需要一個新的連接池?
原因如下:
- Commons DBCP 1.x 是單線程。為了線程安全,在對象分配或對象返回的短期內,Commons 鎖定了全部池。但注意這并不適用于 Commons DBCP 2.x。
- Commons DBCP 1.x 可能會變得很慢。當邏輯 CPU 數目增長,或者試圖借出或歸還對象的并發線程增加時,性能就會受到影響。高并發系統受到的影響會更為顯著。注意這并不適用于 Commons DBCP 2.x。
- Commons DBCP 擁有 60 多個類。tomcat-jdbc-pool 核心只有 8 個類。因此為了未來需求變更著想,肯定需要更少的改動。我們真正需要的只是連接池本身,其余的只是附屬。
- Commons DBCP 使用靜態接口,因此對于指定版本的 JRE,只能采用正確版本的 DBCP,否則就會出現 NoSuchMethodException 異常。
- 當DBCP 可以用其他更簡便的實現來替代時,實在不值得重寫那 60 個類。
- Tomcat JDBC 連接池無需為庫本身添加額外線程,就能獲取異步獲取連接。
- Tomcat JDBC 連接池是 Tomcat 的一個模塊,依靠 Tomcat JULI 這個簡化了的日志架構。
- 使用 javax.sql.PooledConnection 接口獲取底層連接。
- 防止饑餓。如果池變空,線程將等待一個連接。當連接返回時,池就將喚醒正確的等待線程。大多數連接池只會一直維持饑餓狀態。
Tomcat JDBC 連接池還具有一些其他連接池實現所沒有的特點:
- 支持高并發環境與多核/CPU 系統。
- 接口的動態實現。支持 java.sql 與 java.sql 接口(只要 JDBC 驅動),甚至在利用低版本的 JDK 來編譯時。
- 驗證間隔時間。我們不必每次使用單個連接時都進行驗證,可以在借出或歸還連接時進行驗證,只要不低于我們所設定的間隔時間就行。
- 只執行一次查詢。當與數據庫建立起連接時,只執行一次的可配置查詢。這項功能對會話設置非常有用,因為你可能會想在連接建立的整個時段內都保持會話。
- 能夠配置自定義攔截器。通過自定義攔截器來增強功能。可以使用攔截器來采集查詢統計,緩存會話狀態,重新連接之前失敗的連接,重新查詢,緩存查詢結果,等等。由于可以使用大量的選項,所以這種自定義攔截器也是沒有限制的,跟 java.sql/javax.sql 接口的 JDK 版本沒有任何關系。
- 高性能。后文將舉例展示一些性能差異。
- 極其簡單。它的實現非常簡單,代碼行數與源文件都非常少,這都有賴于從一開始研發它時,就把簡潔當做重中之重。對比一下 c3p0 ,它的源文件超過了 200 個(最近一次統計),而 Tomcat JDBC 核心只有 8 個文件,連接池本身則大約只有這個數目的一半,所以能夠輕易地跟蹤和修改可能出現的 Bug。
- 異步連接獲取。可將連接請求隊列化,系統返回 Future。
- 更好地處理空閑連接。不再簡單粗暴地直接把空閑連接關閉,而是仍然把連接保留在池中,通過更為巧妙的算法控制空閑連接池的規模。
- 可以控制連接應被廢棄的時間:當池滿了即廢棄,或者指定一個池使用容差值,發生超時就進行廢棄處理。
- 通過查詢或語句來重置廢棄連接計時器。允許一個使用了很長時間的連接不因為超時而被廢棄。這一點是通過使用 ResetAbandonedTimer 來實現的。
- 經過指定時間后,關閉連接。與返回池的時間相類似。
- 當連接要被釋放時,獲取 JMX 通知并記錄所有日志。它類似于 removeAbandonedTimeout,但卻不需要采取任何行為,只需要報告信息即可。通過 suspectTimeout 屬性來實現。
- 可以通過 java.sql.Driver、javax.sql.DataSource 或 javax.sql.XADataSource 獲取連接。通過 dataSource 與 dataSourceJNDI 屬性實現這一點。
- 支持 XA 連接。