數據庫連接池實現原理
一、為什麽在連接數據庫時要使用連接池
數據庫連接是一種關鍵的有限的昂貴的資源,這一點在多用戶的網頁應用程序中體現得尤為突出。 一個數據庫連接對象均對應一個物理數據庫連接,每次操作都打開一個物理連接,使用完都關閉連接,這樣造成系統的 性能低下。 數據庫連接池的解決方案是在應用程序啟動時建立足夠的數據庫連接,並講這些連接組成一個連接池(簡單說:在一個“池”裏放了好多半成品的數據庫聯接對象),由應用程序動態地對池中的連接進行申請、使用和釋放。對於多於連接池中連接數的並發請求,應該在請求隊列中排隊等待。並且應用程序可以根據池中連接的使用率,動態增加或減少池中的連接數。 連接池技術盡可能多地重用了消耗內存地資源,大大節省了內存,提高了服務器地服務效率,能夠支持更多的客戶服務。通過使用連接池,將大大提高程序運行效率,同時,我們可以通過其自身的管理機制來監視數據庫連接的數量、使用情況等。
二、數據庫連接池的基本原理
數據庫連接池的基本思想就是為數據庫連接 建立一個“緩沖池”。預先在緩沖池中放入一定數量的連接,當需要建立數據庫連接時,只需從“緩沖池”中取出一個,使用完畢之後再放回去。我們可以通過設定 連接池最大連接數來防止系統無盡的與數據庫連接。更為重要的是我們可以通過連接池的管理機制監視數據庫的連接的數量?使用情況,為系統開發?測試及性能調 整提供依據。
三、數據庫連接池的工作原理
連接池的工作原理主要由三部分組成,分別為連接池的建立、連接池中連接的使用管理、連接池的關閉。
第一、連接池的建立。一般在系統初始化時,連接池會根據系統配置建立,並在池中創建了幾個連接對象,以便使用時能從連接池中獲取。連接池中的連接不能隨意創建和關閉,這樣避免了連接隨意建立和關閉造成的系統開銷。Java中提供了很多容器類可以方便的構建連接池,例如Vector、Stack等。
第二、連接池的管理。連接池管理策略是連接池機制的核心,連接池內連接的分配和釋放對系統的性能有很大的影響。其管理策略是:
當客戶請求數據庫連接時,首先查看連接池中是否有空閑連接,如果存在空閑連接,則將連接分配給客戶使用;如果沒有空閑連接,則查看當前所開的連接數是否已經達到最大連接數,如果沒達到就重新創建一個連接給請求的客戶;如果達到就按設定的最大等待時間進行等待,如果超出最大等待時間,則拋出異常給客戶。
當客戶釋放數據庫連接時,先判斷該連接的引用次數是否超過了規定值,如果超過就從連接池中刪除該連接,否則保留為其他客戶服務。
該策略保證了數據庫連接的有效復用,避免頻繁的建立、釋放連接所帶來的系統資源開銷。
第三、連接池的關閉。當應用程序退出時,關閉連接池中所有的連接,釋放連接池相關的資源,該過程正好與創建相反。
四、連接池關鍵問題分析
1、並發問題
為了使連接管理服務具有最大的通用性,必須考慮多線程環境,即並發問題。這個問題相對比較好解決,因為Java語言自身提供了對並發管理的支 持,使用synchronized關鍵字即可確保線程是同步的。使用方法為直接在類方法前面加上synchronized關鍵字,如:
public synchronized Connection getConnection()
2、多數據庫服務器和多用戶
對於大型的企業級應用,常常需要同時連接不同的數據庫(如連接Oracle和Sybase)。如何連接不同的數據庫呢?我們采用的策略是:設計 一個符合單例模式的連接池管理類,在連接池管理類的唯一實例被創建時讀取一個資源文件,其中資源文件中存放著多個數據庫的url地址()?用戶名()?密 碼()等信息。如 tx.url=172.21.15.123:5000/tx_it,tx.user=yang,tx.password=yang321。根據資源文件提 供的信息,創建多個連接池類的實例,每一個實例都是一個特定數據庫的連接池。連接池管理類實例為每個連接池實例取一個名字,通過不同的名字來管理不同的連 接池。
對於同一個數據庫有多個用戶使用不同的名稱和密碼訪問的情況,也可以通過資源文件處理,即在資源文件中設置多個具有相同url地址,但具有不同用戶名和密碼的數據庫連接信息。
3、事務處理
我們知道,事務具有原子性,此時要求對數據庫的操作符合“ALL-ALL-NOTHING”原則,即對於一組SQL語句要麽全做,要麽全不做。
在Java語言中,Connection類本身提供了對事務的支持,可以通過設置Connection的AutoCommit屬性為 false,然後顯式的調用commit或rollback方法來實現。但要高效的進行Connection復用,就必須提供相應的事務支持機制。可采用 每一個事務獨占一個連接來實現,這種方法可以大大降低事務管理的復雜性。
4、連接池的分配與釋放
連接池的分配與釋放,對系統的性能有很大的影響。合理的分配與釋放,可以提高連接的復用度,從而降低建立新連接的開銷,同時還可以加快用戶的訪問速度。
對於連接的管理可使用空閑池。即把已經創建但尚未分配出去的連接按創建時間存放到一個空閑池中。每當用戶請求一個連接時,系統首先檢查空閑池內 有沒有空閑連接。如果有就把建立時間最長(通過容器的順序存放實現)的那個連接分配給他(實際是先做連接是否有效的判斷,如果可用就分配給用戶,如不可用 就把這個連接從空閑池刪掉,重新檢測空閑池是否還有連接);如果沒有則檢查當前所開連接池是否達到連接池所允許的最大連接數(maxConn),如果沒有 達到,就新建一個連接,如果已經達到,就等待一定的時間(timeout)。如果在等待的時間內有連接被釋放出來就可以把這個連接分配給等待的用戶,如果 等待時間超過預定時間timeout,則返回空值(null)。系統對已經分配出去正在使用的連接只做計數,當使用完後再返還給空閑池。對於空閑連接的狀 態,可開辟專門的線程定時檢測,這樣會花費一定的系統開銷,但可以保證較快的響應速度。也可采取不開辟專門線程,只是在分配前檢測的方法。
5、連接池的配置與維護
連接池中到底應該放置多少連接,才能使系統的性能最佳?系統可采取設置最小連接數(minConn)和最大連接數(maxConn)來控制連接 池中的連接。最小連接數是系統啟動時連接池所創建的連接數。如果創建過多,則系統啟動就慢,但創建後系統的響應速度會很快;如果創建過少,則系統啟動的很 快,響應起來卻慢。這樣,可以在開發時,設置較小的最小連接數,開發起來會快,而在系統實際使用時設置較大的,因為這樣對訪問客戶來說速度會快些。最大連 接數是連接池中允許連接的最大數目,具體設置多少,要看系統的訪問量,可通過反復測試,找到最佳點。
如何確保連接池中的最小連接數呢?有動態和靜態兩種策略。動態即每隔一定時間就對連接池進行檢測,如果發現連接數量小於最小連接數,則補充相應數量的新連接,以保證連接池的正常運轉。靜態是發現空閑連接不夠時再去檢查。
程序開發過程中,存在很多問題:
首先,每一次web請求都要建立一次數據庫連接。建立連接是一個費時的活動,每次都得花費0.05s~1s的時間,而且系統還要分配內存資源。這個時間對於一次或幾次數據庫操作,或許感覺不出系統有多大的開銷。
可是對於現在的web應用,尤其是大型電子商務網站,同時有幾百人甚至幾千人在線是很正常的事。在這種情況下,頻繁的進行數據庫連接操作勢必占用很多的系統資源,網站的響應速度必定下降,嚴重的甚至會造成服務器的崩潰。不是危言聳聽,這就是制約某些電子商務網站發展的技術瓶頸問題。其次,對於每一次數據庫連接,使用完後都得斷開。否則,如果程序出現異常而未能關閉,將會導致數據庫系統中的內存泄漏,最終將不得不重啟數據庫
通過上面的分析,我們可以看出來,“數據庫連接”是一種稀缺的資源,為了保障網站的正常使用,應該對其進行妥善管理。其實我們查詢完數據庫後,如果不關閉連接,而是暫時存放起來,當別人使用時,把這個連接給他們使用。就避免了一次建立數據庫連接和斷開的操作時間消耗。
數據庫連接池的基本思想:就是為數據庫連接建立一個“緩沖池”。預先在緩沖池中放入一定數量的連接,當需要建立數據庫連接時,只需從“緩沖池”中取出一個,使用完畢之後再放回去。我們可以通過設定連接池最大連接數來防止系統無盡的與數據庫連接
創建數據庫連接池大概有3個步驟:
① 創建ConnectionPool實例,並初始化創建10個連接,保存在Vector中(線程安全)
② 實現getConnection()從連接庫中獲取一個可用的連接
③ returnConnection(conn) 提供將連接放回連接池中方法
五、連接池實現代碼(java)
[java] view plain copy- package book.util;
- import java.sql.Connection;
- import java.sql.DatabaseMetaData;
- import java.sql.Date;
- import java.sql.Driver;
- import java.sql.DriverManager;
- import java.sql.PreparedStatement;
- import java.sql.ResultSet;
- import java.sql.SQLException;
- import java.sql.Statement;
- import java.util.Vector;
- public class Pool {
- public static void main(String[] args) {
- Pool pool = new Pool("com.microsoft.sqlserver.jdbc.SQLServerDriver","jdbc:sqlserver://localhost:1433;DataBaseName=Book","sa","aaaaaa");
- try {
- pool.createConnections(4);
- } catch (SQLException e) {
- e.printStackTrace();
- }
- Connection conn = pool.getConnection();
- try {
- String sql = "select * from allbook";
- PreparedStatement ps;
- ps = conn.prepareStatement(sql);
- ResultSet rs=ps.executeQuery();
- while(rs.next()){
- System.out.println(rs.getString("BOOKNAME"));
- }
- } catch (SQLException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }finally{
- pool.returnConnection(conn);
- }
- //long startTime=System.currentTimeMillis();
- //long endTime=System.currentTimeMillis();
- //System.out.println("程序運行時間: "+(endTime-startTime)+"ms");
- }
- private String jdbcDriver = "";//數據庫驅動
- private String dbUrl = "";//數據庫url
- private String dbUsername = "";//數據庫用戶名
- private String dbPassword = "";//數據庫密碼
- private String testTable = "";
- private int initialConnectionsNum = 10;//連接池初始連接數
- private int maxConnectionsNum = 50;//連接池最大連接數
- private int incrementalConnections = 5;//每次動態添加的連接數
- private Vector<PooledConnection> connections = null;//向量,存放連接池中的連接,初始為空
- /*無參構造函數*/
- public Pool()
- {}
- /*帶參數的構造函數
- * 初始化數據庫驅動、數據庫url、數據庫用戶名、數據庫密碼、測試表
- * */
- public Pool(String driver, String url, String name, String pass)
- {
- this.jdbcDriver = driver;
- this.dbUrl = url;
- this.dbUsername = name;
- this.dbPassword = pass;
- //this.testTable = table;
- try {
- this.createPool();
- } catch (InstantiationException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (IllegalAccessException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (ClassNotFoundException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (SQLException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- /*函數,創建連接池*/
- public synchronized void createPool()
- throws InstantiationException, IllegalAccessException,
- ClassNotFoundException, SQLException
- {
- /*確保連接池為創建,如果已經創建,則保存連接的向量不為空
- * */
- if (this.connections != null)
- {
- return ;
- }
- //驅動器實例化
- Driver driver = (Driver)(Class.forName(this.jdbcDriver).newInstance());
- //註冊驅動器
- DriverManager.registerDriver(driver);
- //創建保存連接的向量
- this.connections = new Vector<PooledConnection>();
- //創建數據庫連接
- this.createConnections(this.initialConnectionsNum);
- }
- /*函數,創建數據庫連接
- * */
- private void createConnections (int num) throws SQLException
- {
- /*循環創建連接
- * 需要首先檢查當前連接數是否已經超出連接池最大連接數
- * */
- for (int i = 0; i < num; ++i)
- {
- //檢查
- if (this.connections.size() >= this.maxConnectionsNum)
- {
- return;
- }
- //創建連接
- this.connections.addElement
- (new PooledConnection(newConnection()));
- }
- }
- /*函數,創建一個數據庫連接*/
- private Connection newConnection() throws SQLException
- {
- /*創建連接*/
- Connection con = DriverManager.getConnection(this.dbUrl,
- this.dbUsername, this.dbPassword);
- /*如果是第一次創建連接,則檢查所連接的數據庫的允許最大連接數是否小於
- * 我們所設定的最大連接數*/
- if (this.connections.size() == 0)
- {
- DatabaseMetaData metadata = con.getMetaData();
- //得到數據庫最大連接數
- int dbMaxConnectionsNum = metadata.getMaxConnections();
- //如果數據庫最大連接數更小,則更改我們所設定的連接池最大連接數
- if (dbMaxConnectionsNum > 0
- && this.maxConnectionsNum > dbMaxConnectionsNum)
- {
- this.maxConnectionsNum = dbMaxConnectionsNum;
- }
- }
- return con;
- }
- /*函數,得到一個可用連接
- * */
- public synchronized Connection getConnection ()
- {
- Connection con = null;
- /*檢查連接池是否已經建立*/
- if (this.connections == null)
- {
- return con;
- }
- //得到一個可用連接
- try {
- con = this.getFreeConnection();
- } catch (SQLException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- //如果未找到合適連接,循環等待、查找,知道找到合適連接
- while(con == null)
- {
- this.wait(30);
- try {
- con = this.getFreeConnection();
- } catch (SQLException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- return con;
- }
- /*函數,得到一個可用連接*/
- private Connection getFreeConnection() throws SQLException
- {
- Connection con = null;
- //查找一個可用連接
- con = this.findFreeConnection();
- //如果未找到可用連接,就建立一些新的連接,再次查找
- if (con == null)
- {
- this.createConnections(this.incrementalConnections);
- //再次查找
- con = this.findFreeConnection();
- }
- return con;
- }
- /*函數,從現有連接中查找一個可用連接
- * 在現有的連接中(向量connections中)找到一個空閑連接,
- * 並測試這個鏈接是否可用,若不可用則重新建立連接,替換原來的連接*/
- private Connection findFreeConnection () throws SQLException
- {
- Connection con = null;
- for (int i = 0; i < this.connections.size(); ++i)
- {
- PooledConnection pol = (PooledConnection)this.connections.get(i);
- if (!pol.isBusy())
- {
- /*如果此鏈接未被使用,則返回這個連接並,設置正在使用標誌*/
- con = pol.getCon();
- pol.setBusy(true);
- /*測試連接是否可用*/
- if (!this.testCon(con))
- {
- con = this.newConnection();
- pol.setCon(con);
- }
- break;
- }
- }
- return con;
- }
- /*函數,測試連接是否可用
- * */
- private boolean testCon (Connection con)
- {
- boolean useable = true;
- try
- {
- Statement st = con.createStatement();
- ResultSet rs = st.executeQuery("select count(*) from " + this.testTable);
- rs.next();
- }
- catch(SQLException e)
- {
- /*上面拋出異常,連接不可用,關閉*/
- useable = false;
- this.closeConnection(con);
- }
- return useable;
- }
- /*函數,將使用完畢的連接放回連接池中
- * */
- public void returnConnection(Connection con)
- {
- /*確保連接池存在*/
- if (this.connections == null)
- {
- return ;
- }
- for (int i = 0; i < this.connections.size(); ++i)
- {
- PooledConnection pool = this.connections.get(i);
- //找到相應連接,設置正在使用標誌為false
- if (con == pool.getCon())
- {
- pool.setBusy(false);
- }
- }
- }
- /*函數,刷新連接池中的連接*/
- public synchronized void refreshConneciontPool () throws SQLException
- {
- /*確保連接池存在*/
- if (this.connections == null)
- {
- return ;
- }
- for (int i = 0; i < this.connections.size(); ++i)
- {
- PooledConnection pool = this.connections.get(i);
- if (pool.isBusy())
- {
- this.wait(5000);
- }
- this.closeConnection(pool.getCon());
- pool.setCon(this.newConnection());
- pool.setBusy(false);
- }
- }
- /*函數,關閉連接池*/
- public void closeConnectionPool()
- {
- /*確保連接池存在*/
- if (this.connections == null)
- {
- return ;
- }
- for (int i = 0; i < this.connections.size(); ++i)
- {
- PooledConnection pool = this.connections.get(i);
- if (pool.isBusy())
- {
- this.wait(5000);
- }
- this.closeConnection(pool.getCon());
- this.connections.remove(i);
- }
- this.connections = null;
- }
- /*函數,暫時無可用連接,進入等待隊列等待m秒,再試
- * */
- private void wait(int mSecond)
- {
- try {
- Thread.sleep(mSecond);
- } catch (InterruptedException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- /**
- * @return the jdbcDriver
- */
- public String getJdbcDriver() {
- return jdbcDriver;
- }
- /**
- * @param jdbcDriver the jdbcDriver to set
- */
- public void setJdbcDriver(String jdbcDriver) {
- this.jdbcDriver = jdbcDriver;
- }
- /**
- * @return the dbUrl
- */
- public String getDbUrl() {
- return dbUrl;
- }
- /**
- * @param dbUrl the dbUrl to set
- */
- public void setDbUrl(String dbUrl) {
- this.dbUrl = dbUrl;
- }
- /**
- * @return the dbUsername
- */
- public String getDbUsername() {
- return dbUsername;
- }
- /**
- * @param dbUsername the dbUsername to set
- */
- public void setDbUsername(String dbUsername) {
- this.dbUsername = dbUsername;
- }
- /**
- * @return the dbPassword
- */
- public String getDbPassword() {
- return dbPassword;
- }
- /**
- * @param dbPassword the dbPassword to set
- */
- public void setDbPassword(String dbPassword) {
- this.dbPassword = dbPassword;
- }
- /**
- * @return the testTable
- */
- public String getTestTable() {
- return testTable;
- }
- /**
- * @param testTable the testTable to set
- */
- public void setTestTable(String testTable) {
- this.testTable = testTable;
- }
- /**
- * @return the initialConnectionsNum
- */
- public int getInitialConnectionsNum() {
- return initialConnectionsNum;
- }
- /**
- * @param initialConnectionsNum the initialConnectionsNum to set
- */
- public void setInitialConnectionsNum(int initialConnectionsNum) {
- this.initialConnectionsNum = initialConnectionsNum;
- }
- /**
- * @return the maxConnectionsNum
- */
- public int getMaxConnectionsNum() {
- return maxConnectionsNum;
- }
- /**
- * @param maxConnectionsNum the maxConnectionsNum to set
- */
- public void setMaxConnectionsNum(int maxConnectionsNum) {
- this.maxConnectionsNum = maxConnectionsNum;
- }
- /**
- * @return the incrementalConnections
- */
- public int getIncrementalConnections() {
- return incrementalConnections;
- }
- /**
- * @param incrementalConnections the incrementalConnections to set
- */
- public void setIncrementalConnections(int incrementalConnections) {
- this.incrementalConnections = incrementalConnections;
- }
- /**
- * @return the connections
- */
- public Vector<PooledConnection> getConnections() {
- return connections;
- }
- /**
- * @param connections the connections to set
- */
- public void setConnections(Vector<PooledConnection> connections) {
- this.connections = connections;
- }
- /*函數,連接使用完畢,關閉連接*/
- private void closeConnection (Connection con)
- {
- try
- {
- con.close();
- }
- catch(SQLException e)
- {
- e.printStackTrace();
- }
- }
- /*內部使用的保存數據庫連接的類
- * 兩個成員變量:連接、是否正在使用*/
- class PooledConnection
- {
- private Connection con = null;//連接
- private boolean busy = false;//是否正在使用,默認為非
- /*構造函數*/
- public PooledConnection(Connection con)
- {
- this.con = con;
- }
- /**
- * @return the con
- */
- public Connection getCon() {
- return con;
- }
- /**
- * @param con the con to set
- */
- public void setCon(Connection con) {
- this.con = con;
- }
- /**
- * @return the busy
- */
- public boolean isBusy() {
- return busy;
- }
- /**
- * @param busy the busy to set
- */
- public void setBusy(boolean busy) {
- this.busy = busy;
- }
- }
- }
數據庫連接池實現原理