91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

數據庫連接池的原理是什么

發布時間:2021-12-03 15:59:46 來源:億速云 閱讀:226 作者:柒染 欄目:大數據

今天就跟大家聊聊有關數據庫連接池的原理是什么,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結了以下內容,希望大家根據這篇文章可以有所收獲。

背景介紹

數據庫連接池和線程池等池技術存在的意義都是為了解決資源的重復利用問題。在計算機里,創建一個新的資源往往開銷是非常大的。而池技術可以統一分配,管理某一類資源,它允許我們的程序可以重復的使用這個資源,只有在極端情況下(比如連接池滿)才會創建新的資源。

數據庫連接這種資源尤其昂貴,它的創建開銷很大,大量的創建連接和釋放操作對程序的影響非常明顯。

數據庫連接池正是針對這個問題提出來的。

數據庫連接池的原理是什么

 

實現原理

需要注意的是,我們下面提供的幾種實現方式都是基于簡單的原型,目的是帶你了解連接池實現的一些基本原理。真實的數據庫連接池技術需要考慮更多復雜的細節。

所以下面這些代碼都是不能在生產上直接使用的

實現的時候會用到java.sql.Connection,由于這個只是一個接口無法創建實例,為了演示方便,我繼承這個接口寫了一個簡單的測試類,只是在commit方法里加了延時模擬提交。

數據庫連接池的原理是什么

 

實現方式1

很容易馬上想到的一種方案,我們用一個map存放連接對象,需要的時候從map里拿來用就可以了。

數據庫連接池的原理是什么

這里需要主要,盡管我們使用了線程安全的ConcurrentHashMap來存放連接資源,getConnection方法依然要加上synchronized關鍵字來避免并發問題。這一點是最容易忽略的。

試想一下,假設在某個場景下,我們希望某個應用的多個線程共享連接資源。

假設有2個線程同時執行到了pool.containsKey(key),然后都返回false,那這兩個線程都會創建連接。雖然ConcurrentHashMap的put方法只會加入其中一個,但還是生成了1個多余的連接。

原因在于,盡管ConcurrentHashMap本身每個操作都是線程安全的,但是當這些操作組合在一起使用的時候,就無法保證原子性了,所以有可能帶來并發的問題。

這里友情提示下,面試經常會遇到這個考點哦。

 

實現方式2

第二種實現方式是在1的基礎上進行的優化。1的方案有個問題就是每次訪問getConnection都要加鎖,釋放鎖,效率比較低。

第二種方案是利用java并發包里的Future機制來解決并發場景創建多余連接的問題。

數據庫連接池的原理是什么

我們來捋一捋這個實現會不會有并發的問題。假設兩個線程同時進入else分支,在代碼的28行ConcurrentHashMap可以確保只有一個線程會執行,也就是只會加入一個task。其它的線程都不會加入成功。

所以只有一個線程connectionFutureTask == null,這個線程開始異步執行創建連接的任務,而其它的線程則會調用FutureTask的get方法直接獲取結果。

 

實現方式3

1和2的實現方式還存在一個問題, 多個線程獲取到的其實同一個連接。這種方案在某些場景下是不允許的。比如spring數據庫的事務管理器對于每個事務的處理線程都要求獨立的連接資源。

下面的方案基于鏈表結構,有比較完整的獲取,釋放的操作,不同的線程可以拿到獨立的連接資源。

數據庫連接池的原理是什么

注意到這個方案我們在獲取連接的時候引入了超時時間,如果該方法能夠在一段時間內獲取到結果,那么將結果立刻返回,反之,超時返回默認結果。

 

druid連接池的實現原理

了解了實現連接池的大概思路,我們可以來繼續學習下市面上比較成熟的連接池產品。這其中阿里巴巴開源的druid開源連接池就是一個代表。

Druid作為java領域最好的連接池技術之一,連接池本身只是它的一部分功能。除此之外,它還還要配套的監控功能。當然這個不是我們本文的重點。

先來看看在代碼中如何使用Druid連接池,

數據庫連接池的原理是什么

所以繼續深入到是DataSource里的getConnection方法,

數據庫連接池的原理是什么

init方法主要的功能是根據配置文件初始化連接池,它內部會生成一些真正的物理連接然后放入一個數組里。當然這個方法要保證只會被調用一次。

繼續往下看,最終會調用到getConnectionInternal這個私有方法,

數據庫連接池的原理是什么

紅色圈出的部分是核心,根據傳入的等待時間走不同的分支,我們來看看takeLast方法。

數據庫連接池的原理是什么

代碼邏輯也比較清楚,poolCount是連接池的目前的可用連接數量。

如果為0,就通過emptySignal喚醒生產者線程創建新的連接,同時當前線程掛起等待notEmpty的信號。notEmptyWaitCount維護的就是正在等待的消費者數量。

如果不為0,就從數組中取出最后一個連接返回。有人可能會有疑問,這里返回的是DruidConnectionHolder,不是Connection啊?

其實看下前者的定義你就明白了,

數據庫連接池的原理是什么

DruidConnectionHolder封裝了Connection以及連接的datasource信息,還有多個statement等,方面進行統一管理。

看完上述內容,你們對數據庫連接池的原理是什么有進一步的了解嗎?如果還想了解更多知識或者相關內容,請關注億速云行業資訊頻道,感謝大家的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

新建县| 灵丘县| 湖北省| 安阳县| 台安县| 黄浦区| 新和县| 华亭县| 金昌市| 雅江县| 隆昌县| 曲阜市| 南昌市| 梨树县| 南开区| 永平县| 崇左市| 吕梁市| 西青区| 天长市| 青州市| 策勒县| 碌曲县| 张家川| 青田县| 临桂县| 乐昌市| 阳信县| 驻马店市| 大方县| 当雄县| 五大连池市| 铅山县| 芒康县| 凉山| 军事| 嘉黎县| 来宾市| 浦城县| 晴隆县| 蒲江县|