單例模式是限制了一個類只能有一個實例,對象池模式則是限制一個類實例的個數。對象池類就像是一個對象管理員,它以Static列表(也就是裝對象的池子)的形式存存儲某個實例數受限的類的實例,每個實例還要加一個標記,標記該實例是否被佔用。當類初始化的時候,這個對象池就被初始化了,實例就被建立出來。而後,用戶能夠向這個類索取實例,若是池中全部的實例都已經被佔用了,那麼拋出異常。用戶用完之後,還要把實例「還」回來,即釋放佔用。對象池類的成員應該都是靜態的。用戶也不該該能訪問池子裏裝着的對象的構造函數,以防用戶繞開對象池建立實例。書上說這個模式會用在數據庫鏈接的管理上。好比,每一個用戶的鏈接數是有限的,這樣每一個鏈接就是一個池子裏的一個對象,「鏈接池」類就能夠控制鏈接數了。java
Java對象的生命週期分析
Java對象的生命週期大體包括三個階段:對象的建立,對象的使用,對象的清除。所以,對象的生命週期長度可用以下的表達式表示:T = T1 + T2 +T3。其中T1表示對象的建立時間,T2表示對象的使用時間,而T3則表示其清除時間。由此,咱們能夠看出,只有T2是真正有效的時間,而T一、T3則是對象自己的開銷。下面再看看T一、T3在對象的整個生命週期中所佔的比例。
咱們知道,Java對象是經過構造函數來建立的,在這一過程當中,該構造函數鏈中的全部構造函數也都會被自動調用。另外,默認狀況下,調用類的構造函數時,Java會把變量初始化成肯定的值:全部的對象被設置成null,整數變量(byte、short、int、long)設置成0,float和double變量設置成0.0,邏輯值設置成false。因此用new關鍵字來新建一個對象的時間開銷是很大的,如表1所示。
表1 一些操做所耗費時間的對照表程序員
運算操做數據庫 |
示例apache |
標準化時間數組 |
本地賦值緩存 |
i = n函數 |
1.0性能 |
實例賦值測試 |
this.i = nthis |
1.2 |
方法調用 |
Funct() |
5.9 |
新建對象 |
New Object() |
980 |
新建數組 |
New int[10] |
3100 |
從表1能夠看出,新建一個對象須要980個單位的時間,是本地賦值時間的980倍,是方法調用時間的166倍,而若新建一個數組所花費的時間就更多了。
再看清除對象的過程。咱們知道,Java語言的一個優點,就是Java程序員勿需再像C/C++程序員那樣,顯式地釋放對象,而由稱爲垃圾收集器(Garbage Collector)的自動內存管理系統,定時或在內存凸現出不足時,自動回收垃圾對象所佔的內存。凡事有利總也有弊,這雖然爲Java程序設計者提供了極大的方便,但同時它也帶來了較大的性能開銷。這種開銷包括兩方面,首先是對象管理開銷,GC爲了可以正確釋放對象,它必須監控每個對象的運行狀態,包括對象的申請、引用、被引用、賦值等。其次,在GC開始回收「垃圾」對象時,系統會暫停應用程序的執行,而獨自佔用CPU。
所以,若是要改善應用程序的性能,一方面應儘可能減小建立新對象的次數;同時,還應儘可能減小T一、T3的時間,而這些都可以經過對象池技術來實現。
對象池技術的基本原理
對象池技術基本原理的核心有兩點:緩存和共享,即對於那些被頻繁使用的對象,在使用完後,不當即將它們釋放,而是將它們緩存起來,以供後續的應用程序重複使用,從而減小建立對象和釋放對象的次數,進而改善應用程序的性能。事實上,因爲對象池技術將對象限制在必定的數量,也有效地減小了應用程序內存上的開銷。
對象池使用的基本思路是:
將用過的對象保存起來,等下一次須要這種對象的時候,再拿出來重複使用,從而在必定程度上減小頻繁建立對象所形成的開銷。 並不是全部對象都適合拿來池化――由於維護對象池也要形成必定開銷。對生成時開銷不大的對象進行池化,反而可能會出現「維護對象池的開銷」大於「生成新對象的開銷」,從而使性能下降的狀況。可是對於生成時開銷可觀的對象,池化技術就是提升性能的有效策略了。下面是構建對象池的一個例子:
commons-pool提供了一套很好用的對象池組件。使用也很簡單,不過對一些簡單的對象使用對象池就不必了。
ObjectPool定義了一個簡單的池化接口,有三個對應實現
GenericObjectPool:實現了可配置的後進先出或先進先出(LIFO/FIFO)行爲,默認是做爲一個後進先出隊列,這意味當對象池中有可用的空閒對象時,borrowObject 將返回最近的對象實例,若是將lifo 屬性設置爲false,則按FIFO行爲返回對象實例。
StackObjectPool :實現了後進先出(LIFO)行爲。
SoftReferenceObjectPool: 實現了後進先出(LIFO)行爲。另外,對象池還在SoftReference 中保存了每一個對象引用,容許垃圾收集器針對內存須要回收對象。
KeyedObjectPool定義了一個以任意的key訪問對象的接口(能夠池化對種對象),有兩種對應實現。
GenericKeyedObjectPool :實現了先進先出(FIFO)行爲。
StackKeyedObjectPool : 實現了後進先出(LIFO)行爲。
PoolableObjectFactory 定義了池化對象的生命週期方法,咱們可使用它分離被池化的不一樣對象和管理對象的建立,持久,銷燬。
BasePoolableObjectFactory這個實現PoolableObjectFactory 接口的一個抽象類,咱們可用擴展它實現本身的池化工廠。
一個對象池使用的簡單例子: