java 對象池 博客_Java對象池技術的原理及其實現的小結

一起學習

Java對象的生命周期大致包括三個階段:對象的創建,對象的使用,對象的清除。因此,對象的生命周期長度可用如下的表達式表示:T = T1 T2 T3。其中T1表示對象的創建時間,T2表示對象的使用時間,而T3則表示其清除時間。由此,我們可以看出,只有T2是真正有效的時間,而T1、T3則 是對象本身的開銷。下面再看看T1、T3在對象的整個生命周期中所占的比例。

我們知道,Java對象是通過構造函數來創建的,在這一過程中,該構造函數鏈中的所有構造函數也都會被自動調用。另外,默認情況下,調用類的構造函數 時,Java會把變量初始化成確定的值:所有的對象被設置成null,整數變量(byte、short、int、long)設置成0,float和 double變量設置成0.0,邏輯值設置成false。所以用new關鍵字來新建一個對象的時間開銷是很大的,如表1所示。

表1 一些操作所耗費時間的對照表

運算操作 示例 標準化時間

本地賦值 i = n 1.0

實例賦值 this.i = n 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。

因此,如果要改善應用程序的性能,一方面應盡量減少創建新對象的次數;同時,還應盡量減少T1、T3的時間,而這些均可以通過對象池技術來實現。

對象池技術的基本原理

對象池技術基本原理的核心有兩點:緩存和共享,即對于那些被頻繁使用的對象,在使用完后,不立即將它們釋放,而是將它們緩存起來,以供后續的應用程序重 復使用,從而減少創建對象和釋放對象的次數,進而改善應用程序的性能。事實上,由于對象池技術將對象限制在一定的數量,也有效地減少了應用程序內存上的開 銷。

實現一個對象池,一般會涉及到如下的類:

1)對象池工廠(ObjectPoolFactory)類

該類主要用于管理相同類型和設置的對象池(ObjectPool),它一般包含如下兩個方法:

·createPool:用于創建特定類型和設置的對象池;

·destroyPool:用于釋放指定的對象池;

同時為保證ObjectPoolFactory的單一實例,可以采用Singleton設計模式,見下述getInstance方法的實現:

public static ObjectPoolFactory getInstance() {

if (poolFactory == null) {

poolFactory = new ObjectPoolFactory();

}

return poolFactory;

}

2)參數對象(ParameterObject)類

該類主要用于封裝所創建對象池的一些屬性參數,如池中可存放對象的數目的最大值(maxCount)、最小值(minCount)等。

3)對象池(ObjectPool)類

用于管理要被池化對象的借出和歸還,并通知PoolableObjectFactory完成相應的工作。它一般包含如下兩個方法:

·getObject:用于從池中借出對象;

·returnObject:將池化對象返回到池中,并通知所有處于等待狀態的線程;

4)池化對象工廠(PoolableObjectFactory)類

該類主要負責管理池化對象的生命周期,就簡單來說,一般包括對象的創建及銷毀。該類同ObjectPoolFactory一樣,也可將其實現為單實例。

通用對象池的實現

對象池的構造和管理可以按照多種方式實現。最靈活的方式是將池化對象的Class類型在對象池之外指定,即在ObjectPoolFactory類創建對象池時,動態指定該對象池所池化對象的Class類型,其實現代碼如下:

. . .

public ObjectPool createPool(ParameterObject paraObj,Class clsType) {

return new ObjectPool(paraObj, clsType);

}

. . .

其中,paraObj參數用于指定對象池的特征屬性,clsType參數則指定了該對象池所存放對象的類型。對象池(ObjectPool)創建以后,下面就是利用它來管理對象了,具體實現如下:

public class ObjectPool {

private ParameterObject paraObj;//該對象池的屬性參數對象

private Class clsType;//該對象池中所存放對象的類型

private int currentNum = 0; //該對象池當前已創建的對象數目

private Object currentObj;//該對象池當前可以借出的對象

private Vector pool;//用于存放對象的池

public ObjectPool(ParameterObject paraObj, Class clsType) {

this.paraObj = paraObj;

this.clsType = clsType;

pool = new Vector();

}

public Object getObject() {

if (pool.size() < = paraObj.getMinCount()) {

if (currentNum < = paraObj.getMaxCount()) {

//如果當前池中無對象可用,而且已創建的對象數目小于所限制的最大值,就利用

//PoolObjectFactory創建一個新的對象

PoolableObjectFactory objFactory =PoolableObjectFactory.getInstance();

currentObj = objFactory.create Object (clsType);

currentNum ;

} else {

//如果當前池中無對象可用,而且所創建的對象數目已達到所限制的最大值,

//就只能等待其它線程返回對象到池中

synchronized (this) {

try {

wait();

} catch (InterruptedException e) {

System.out.println(e.getMessage());

e.printStackTrace();

}

currentObj = pool.firstElement();

}

}

} else {

//如果當前池中有可用的對象,就直接從池中取出對象

currentObj = pool.firstElement();

}

return currentObj;

}

public void returnObject(Object obj) {

// 確保對象具有正確的類型

if (obj.isInstance(clsType)) {

pool.addElement(obj);

synchronized (this) {

notifyAll();

}

} else {

throw new IllegalArgumentException("該對象池不能存放指定的對象類型");

}

}

}

從上述代碼可以看出,ObjectPool利用一個java.util.Vector作為可擴展的對象池,并通過它的構造函數來指定池化對象的 Class類型及對象池的一些屬性。在有對象返回到對象池時,它將檢查對象的類型是否正確。當對象池里不再有可用對象時,它或者等待已被使用的池化對象返 回池中,或者創建一個新的對象實例。不過,新對象實例的創建并不在ObjectPool類中,而是由PoolableObjectFactory類的 createObject方法來完成的,具體實現如下:

. . .

public Object createObject(Class clsType) {

Object obj = null;

try {

obj = clsType.newInstance();

} catch (Exception e) {

e.printStackTrace();

}

return obj;

}

. . .

這樣,通用對象池的實現就算完成了,下面再看看客戶端(Client)如何來使用它,假定池化對象的Class類型為StringBuffer:

. . .

//創建對象池工廠

ObjectPoolFactory poolFactory = ObjectPoolFactory. getInstance ();

//定義所創建對象池的屬性

ParameterObject paraObj = new ParameterObject(2,1);

//利用對象池工廠,創建一個存放StringBuffer類型對象的對象池

ObjectPool pool = poolFactory.createPool(paraObj,String Buffer.class);

//從池中取出一個StringBuffer對象

StringBuffer buffer = (StringBuffer)pool.getObject();

//使用從池中取出的StringBuffer對象

buffer.append("hello");

System.out.println(buffer.toString());

. . .

可以看出,通用對象池使用起來還是很方便的,不僅可以方便地避免頻繁創建對象的開銷,而且通用程度高。但遺憾的是,由于需要使用大量的類型定型 (cast)操作,再加上一些對Vector類的同步操作,使得它在某些情況下對性能的改進非常有限,尤其對那些創建周期比較短的對象。

專用對象池的實現

由于通用對象池的管理開銷比較大,某種程度上抵消了重用對象所帶來的大部分優勢。為解決該問題,可以采用專用對象池的方法。即對象池所池化對象的 Class類型不是動態指定的,而是預先就已指定。這樣,它在實現上也會較通用對象池簡單些,可以不要ObjectPoolFactory和 PoolableObjectFactory類,而將它們的功能直接融合到ObjectPool類,具體如下(假定被池化對象的Class類型仍為 StringBuffer,而用省略號表示的地方,表示代碼同通用對象池的實現):

public class ObjectPool {

private ParameterObject paraObj;//該對象池的屬性參數對象

private int currentNum = 0; //該對象池當前已創建的對象數目

private StringBuffer currentObj;//該對象池當前可以借出的對象

private Vector pool;//用于存放對象的池

public ObjectPool(ParameterObject paraObj) {

this.paraObj = paraObj;

pool = new Vector();

}

public StringBuffer getObject() {

if (pool.size() < = paraObj.getMinCount()) {

if (currentNum < = paraObj.getMaxCount()) {

currentObj = new StringBuffer();

currentNum ;

}

. . .

}

return currentObj;

}

public void returnObject(Object obj) {

// 確保對象具有正確的類型

if (StringBuffer.isInstance(obj)) {

. . .

}

}

結束語

恰當地使用對象池技術,能有效地改善應用程序的性能。目前,對象池技術已得到廣泛的應用,如對于網絡和數據庫連接這類重量級的對象,一般都會采用對象池技術。但在使用對象池技術時也要注意如下問題:

·并非任何情況下都適合采用對象池技術。基本上,只在重復生成某種對象的操作成為影響性能的關鍵因素的時候,才適合采用對象池技術。而如果進行池化所能帶來的性能提高并不重要的話,還是不采用對象池化技術為佳,以保持代碼的簡明。

·要根據具體情況正確選擇對象池的實現方式。如果是創建一個公用的對象池技術實現包,或需要在程序中動態指定所池化對象的Class類型時,才選擇通用對象池。而大部分情況下,采用專用對象池就可以了。

下載本文示例代碼

Java對象池技術的原理及其實現的小結Java對象池技術的原理及其實現的小結Java對象池技術的原理及其實現的小結Java對象池技術的原理及其實現的小結Java對象池技術的原理及其實現的小結Java對象池技術的原理及其實現的小結Java對象池技術的原理及其實現的小結Java對象池技術的原理及其實現的小結Java對象池技術的原理及其實現的小結Java對象池技術的原理及其實現的小結Java對象池技術的原理及其實現的小結Java對象池技術的原理及其實現的小結

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/259667.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/259667.shtml
英文地址,請注明出處:http://en.pswp.cn/news/259667.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

matlab中gatbx工具箱的添加

1. 從http://crystalgate.shef.ac.uk/code/下載工具箱壓縮包gatbx.zip 2. 解壓gatbx.zip&#xff0c;將其子文件夾genetic放在matlab安裝目錄toolbox文件夾下 3. 在matlab主窗口選擇File -> Set Path&#xff0c; 單擊"Add Folder"按鈕&#xff0c;找到工具箱所在…

C#與數據庫訪問技術總結(十七)

使用DataSet對象訪問數據庫 當對DataSet對象進行操作時&#xff0c;DataSet對象會產生副本&#xff0c;所以對DataSet里的數據進行編輯操作不會直接對數據庫產生影響&#xff0c;而是將DataRow的狀態設置為added、deleted或changed&#xff0c;最終的更新數據源動作將通過DataA…

MySQL數據高級查詢之連接查詢、聯合查詢、子查詢

2019獨角獸企業重金招聘Python工程師標準>>> 一、連接查詢 連接查詢: 將多張表(>2)進行記錄的連接(按照某個指定的條件進行數據拼接)。 連接查詢的意義: 在用戶查看數據的時候,需要顯示的數據來自多張表. 連接查詢: join, 使用方式: 左表 join 右表&#xff1b;左…

Oracle11g解鎖報錯SP2-0306-選項無效

普通用戶登錄isqlplus: (一)在瀏覽器中輸入URL &#xff08;http://localhost:5560/isqlplus&#xff09;。顯示登錄界面 這里只能用普通用戶進行登錄&#xff0c;因為要用sys登錄&#xff0c;必須用sys的DBA身份登錄。所以用普通用戶SCOTT&#xff0c;但是還未解鎖 問題:SP2-0…

java web登錄action_JavaWeb中登陸功能

首先我們要JavaWeb登陸的基本流程&#xff1a;JSP頁面發送請求——>Servlet——>Servlet通過調用方法從數據庫中得到數據并將結果返回頁面我們先建立三個jsp頁面&#xff0c;包括login.jsp(登陸頁面)、index.jsp(顯示登陸成功后的信息)、error.jsp(登錄失敗的頁面)&#…

Android Download Manager用法大全

http://www.trinea.cn/android/android-downloadmanager/ http://www.trinea.cn/android/android-downloadmanager-pro/轉載于:https://www.cnblogs.com/soaringEveryday/articles/4135204.html

Chrome瀏覽器無法觀看視頻,一直提示“adobe flash player 已過期” ?

很多新用戶在安裝了Chrome瀏覽器或者更新過的的時候&#xff0c;經常提示“ adobe flash player 已過期”的問題&#xff0c;反復提示&#xff0c;導致無法觀看視頻。于是從網上也找了很多辦法都沒有解決。這里給大家提供一個最完美的解決方案。經親自測試&#xff0c;完美解決…

關于JVM的垃圾回收GC的一些記錄

目錄 一、JVM內存區域劃分 二、從一個基本問題開始引入垃圾回收 三、GC作用的區域 三、如何確定一個對象是否可以被當成垃圾進行回收 &#xff08;1&#xff09;引用計數法 &#xff08;2&#xff09;可達性分析算法 &#xff08;3&#xff09;引用的類型 &#xff08;3…

同步與互斥

有數據交互的進程之間的關系主要有兩種,同步與互斥.所謂互斥,是指在不同進程之間的若干程序片斷,當某個進程運行其中一個程序片段時,其它進程就不能運行它們之中的任一程序片段,只能等到該進程運行完這個程序片段后才可以運行.所謂同步,是指在不同進程之間的若干程序片斷,它們的…

java 分號 轉義_java – 正則表達式和轉義和未轉義的分隔符

你可以使用正則表達式(?:\\.|[^;\\])*匹配未轉義分號之間的所有文本&#xff1a;List matchList new ArrayList();try {Pattern regex Pattern.compile("(?:\\\\.|[^;\\\\])*");Matcher regexMatcher regex.matcher(subjectString);while (regexMatcher.find())…

OpenCV cv::Mat類

using namespace cv; 1、Mat的聲明&#xff1a; Mat mMat(rows, cols, type); Mat mMat(Size(width,height), type); type指矩陣中元素的類型&#xff0c;可以使CV_8U&#xff08;無符號單字節像素&#xff09;&#xff0c;CV_8S&#xff08;有符號單字節像素&#xff09;&…

ubuntu-E:Encountered a section with no Package: header的解決辦法

ubuntu 11.04 出現了如下錯誤&#xff1a; E:Encountered a section with no Package: header 終端中輸入以下兩條命令&#xff1a; sudo rm /var/lib/apt/lists/* -vfsudo apt-get update 執行完了命令之后&#xff0c;軟件更新器應該會自動要求更新的&#xff0c;更新便是。 …

java new collection_使用Java 8新增的Predicate操作Collection集合

Java 8 起為 Collection 集合新增了一個 removeIf(Predicate filter) 方法&#xff0c;該方法將會批量刪除符合 filter 條件的所有元素。該方法需要一個 Predicate 對象作為參數&#xff0c;Predicate 也是函數式接口&#xff0c;因此可使用 Lambda 表達式作為參數。如下程序示…

codevs1219 騎士遍歷(棋盤DP)

題目描述 Description設有一個n*m的棋盤&#xff08;2≤n≤50&#xff0c;2≤m≤50&#xff09;&#xff0c;如下圖&#xff0c;在棋盤上有一個中國象棋馬。 規定&#xff1a; 1)馬只能走日字 2)馬只能向右跳 問給定起點x1,y1和終點x2,y2&#xff0c;求出馬從x1,y1出發到x2,y2的…

php顯示錯誤

error_reporting(E_ALL);ini_set(display_errors, 1);//將出錯信息輸出到一個文本文件ini_set(error_log, dirname(__FILE__) . /error_log.txt);

java ssh免密登錄_SSH公鑰、私鑰配置(SSH免密碼登錄方式)

1.首先使用想要發起ssh免密訪問的用戶A登錄Linux(簡稱客戶端Linux)2.進入該用戶的家目錄(cd ~)&#xff0c;看是否有.ssh文件夾(linux中以.開頭文件夾是隱藏的&#xff0c;使用ll -a進行查看)&#xff0c;如果沒有則創建(mkdir ~/.ssh)&#xff0c;并修改訪問權限(chmod 700 ~…

最近學到的一點東西

1 rpc 2 socket&#xff0c;工作很多年了&#xff0c;才明白這貨的重要性&#xff0c;不過什么樣的通信方式&#xff0c;最底層的連接得用它 3 sip協議 4調試&#xff0c;一定要確認異常出錯位置&#xff0c;不能亂猜 5 某些詭異的問題&#xff0c;可以依靠在加一層解決&#x…

win8, VS2013 .NET 4.5在哪找svcutil.exe?

我這個糾結呀&#xff0c;公司用win8&#xff0c; .NET 4.5。想做一個很簡單的項目&#xff0c;就是wcf宿主iis&#xff0c;項目根目錄下有aspx文件和svc文件。于是參考了一個博客http://www.cnblogs.com/yjmyzz/archive/2008/08/19/1270961.html&#xff0c;[原創]WCF入門級使…

java tbase_使用Java從firebase獲取數據

我在下面的代碼中遇到了一個問題,該代碼幾乎逐字地從Firebase SDK Java文檔中復制到了工作中.我是一個真正的語言的新手,比如來自PHP和JavaScript的webdev背景的Java.基本上,addListenerForSingleValueEvent沒有觸發以返回數據.我注意到了這一點,因為系統打印輸出沒有觸發,因此…

錯誤內存【讀書筆記】C程序中常見的內存操作有關的典型編程錯誤

題記&#xff1a;寫這篇博客要主是加深自己對錯誤內存的認識和總結實現算法時的一些驗經和訓教&#xff0c;如果有錯誤請指出&#xff0c;萬分感謝。 對C/C程序員來講&#xff0c;內存管理是個不小的挑戰&#xff0c;絕對值得慎之又慎&#xff0c;否則讓由上萬行代碼構成的模塊…