日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

JAVA中用動態(tài)代理類實現(xiàn)記憶功能(二)

瀏覽:146日期:2024-06-23 08:54:26
內容: 通過動態(tài)代理類來創(chuàng)建一個通用的緩存包裝器 上面第二種方法僅有的缺點就是緩存包裝器不能重用,每次我們希望添加一個緩存給某個類,我們就要寫一個特殊的緩存包裝器給目標接口.這是一個很慢,容易出錯的過程. Jdk1.3開始支持動態(tài)代理類: 特別的類能夠在運行期決定實現(xiàn)哪個接口-通常的模式都是,在運行期即決定實現(xiàn)哪個接口.通過這個,我們有可能實現(xiàn)一個通用的緩存包裝器,我們稱它為Memoizer,在運行期決定實現(xiàn)哪個接口.這樣, CachingBinaryDigitsCalculator就是不再需要的.它是這樣被調用的:BinaryDigitsCalculator calculator = new CachingBinaryDigitsCalculator( new PiBinaryDigitsCalculator() );可以通過Memoizer來重寫如下:BinaryDigitsCalculator calculator = (BinaryDigitsCalculator) Memoizer.memoize( new PiBinaryDigitsCalculator() );Memoizer的代碼如下:import java.lang.reflect.InvocationHandler;import java.lang.reflect.InvocationTargetException;import java.lang.reflect.Method;import java.lang.reflect.Proxy;import java.util.Arrays;import java.util.Collections;import java.util.HashMap;import java.util.List;import java.util.Map;public class Memoizer implements InvocationHandler { public static Object memoize(Object object) { return Proxy.newProxyInstance( object.getClass().getClassLoader(), object.getClass().getInterfaces(), new Memoizer(object) ); } private Object object; private Map caches = new HashMap(); private Memoizer(Object object) { this.object = object; } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { if (method.getReturnType().equals(Void.TYPE)) { // Don't cache void methods return invoke(method, args); } else { Map cache = getCache(method); List key = Arrays.asList(args); Object value = cache.get(key); if (value == null && !cache.containsKey(key)) { value = invoke(method, args); cache.put(key, value); } return value; } } private Object invoke(Method method, Object[] args) throws Throwable { try { return method.invoke(object, args); } catch (InvocationTargetException e) { throw e.getTargetException(); } } private synchronized Map getCache(Method m) { Map cache = (Map) caches.get(m); if (cache == null) { cache = Collections.synchronizedMap( new HashMap() ); caches.put(m, cache); } return cache; }} 當調用靜態(tài)方法memoize的時候,將會創(chuàng)建一個新的代理實例-也就是一個java.lang.reflect.proxy的實例.實現(xiàn)了一個接口集.這個接口集由object.getClass().getInterfaces()來決定.每個代理實例包含一個java.lang.reflect.InvocationHandler實例來處理這個代理實例調用的相關方法.在我們的例子里,Memoizer就是一個InvocationHandler實例. 當一個方法在代理實例里被調用,比如, calculateBinaryDigit,那么, Memoizer實例里的invoke方法就會被調用,相關信息會傳給invoke方法,以決定proxy實例調用了哪個方法,包含參數(shù)信息.在我們的例子里,傳入Memoizer的java.lang.Method參數(shù)是calculateBinaryDigit,而參數(shù)信息則是pi需要精確的位數(shù)-整數(shù)n.在這個基礎上,Memoizer能夠進一步進行緩存操作的. 在例子里(caches是一個Hashmap,cache是一個map)里用到的Key,主要是傳入的方法信息:Method對象和參數(shù)對象. 為了實現(xiàn)的簡單與通用性,Memoizer有一個關于cache的HashMap caches,每個method是一個key,對應的value為一個cache.然后把參數(shù)信息轉化成一個List對象,作為cache的Key.使用List是很方便的,同時也可以保證equals()方法,所以能夠保證當且僅當參數(shù)信息完全相同的時候這個List才相等. 一旦一個cache的Key被創(chuàng)建,那么,計算之前都會先查找這個cache,如果找到,則返回cache里的值.否則,如果帶有這些參數(shù)的這個方法沒有被調用過,那么,則會通過invoke來調用這個method.在我們的例子里, 實例PiBinaryDigitsCalculator 里的calculateBinaryDigit方法將會通過invoke被調用.而且計算結果將會被存在cache里.何時使用Memoizer 作為一條通用的規(guī)則,Memoizer能夠在任何需要傳統(tǒng)的cache的時候使用-比如上面提到的例子. 特別地,接口里每個需要使用記憶功能的method需要滿足下面幾條條件:1. 這個method的返回值最好不要每次調用都會改變2. 這個method不要有副效應3. 這個method的參數(shù)是確定的,非mutable的. 顯然,如果每次調用這個method返回值都不同,那么cache就毫無用處了.同樣也是很重要的一點是,因為有副效應的method不會被重復,所以這個method不能有副效應(method自動更新某些狀態(tài)).當然,void方法除外. 同樣,memorize一個帶有未定(mutable)參數(shù)的method是很危險的,因為,要把這些參數(shù)儲存到hashmap里會是很危險的一件事.根據(jù)Map的定義,當這個Map里的key發(fā)生改變,Map是不知道的.所以,當你執(zhí)行了一次這個method之后,相關信息添加進了Map,然后參數(shù)發(fā)生變異(mutate),第二次調用的時候,就會得到錯誤的結果.性能 使用cache的主要目的就是為了提升你的程序的速度.然而,reflection確是眾所周知的低效(在jdk1.4里有所改進,通過reflection調用方法是普通調用速度的1/2,這個比jdk1.3要快40倍).Memoizer主要依靠reflection來調用方法,所以,它看上去并不是一個好的途徑.但是,如果使用cache能給程序速度帶來的提升遠高于reflection對速度的影響,那么,使用Memoizer是值得考慮的. 在我們對PiBinaryDigitsCalculator的測試中,測試環(huán)境為jdk1.4,當n小于10的時候,使不使用cache速度是相當?shù)?但是,當n增大的時候,使用cache的優(yōu)勢就開始顯示出來.所以,經(jīng)常使用PiBinaryDigitsCalculator的用戶,可以考慮使用cache. 不幸的是,唯一測試你的程序是否需要cache的途徑是比較你的程序在兩種情況下的運行效率.盡管如此,因為為一個程序構造一個cache包裝器是很容易的一件事,移除它也是很容易的,下面的建議可以作為一個參考的步驟:1. 選擇需要記憶操作的類2. 運行它3. 如果效率是滿意的,go to 64. 添加memoizer,使用cache5. 如果效率沒有顯著提升,移初memoizer6. 如果需要,重試. 理論上,你需要分析為一個類添加記憶功能對整個系統(tǒng)的影響.只有你自己清楚是否值得添加.有些方法,即使是計算量很大的,但是在這個系統(tǒng)里很少被調用,所以,沒必要為它添加記憶功能.為了保證這個,我開發(fā)了一個更有特點的Memoizer,實現(xiàn)了一個叫做CacheStatistics的接口,你能從它得到cache的數(shù)量以及無效的cache.你可以使用它作為判斷的一個尺度.擴展Memoizer 修改Memoizer類來支持不同的cache策略是很簡單的.一個比較普通的類型就是Least-Recently-Used(LRU)cahce,擁有固定數(shù)量的入口.這個cache確保入口不大于它的最大數(shù)目,如果超過,就摒棄最舊的緩存數(shù)據(jù).也就是,能夠從cache里得到的是新的數(shù)據(jù).一個類可以使用LRU cache來防止一個程序長期保持一個狀態(tài).你可以僅僅傳遞一個參數(shù)給CacheFactory里的memoize方法來選擇你需要的cache類型.下面的例子,LRU cache最多有1000個入口:BinaryDigitsCalculator calculator = (BinaryDigitsCalculator) Memoizer.memoize( new PiBinaryDigitsCalculator(), new LruCacheFactory(1000) ); 即使是這么簡單,Memoizer也應該是java程序員一個有用的工具.參考資源:• Joshua Bloch, Effective Java Programming Language Guide. Addison Wesley Professional (2001). Contains useful advice about optimization. • Patrick Chan, The Java Developers Almanac 1.4. Addison Wesley Professional (2002). Full of Java recipes, including how to write a LRU cache. • Paul Graham, On Lisp. Prentice Hall (1993). How to write a memoizing function in Common Lisp. • Alex Martelli and David Ascher (Eds.), Python Cookbook. O'Reilly (July 2002). Recipe 17.7, Memoizing (Caching) the Return Values of Functions shows you how to memoize in Python. • Documentation for LinkedHashMap in Java SDK 1.4 • Documentation for Proxy in Java SDK 1.4 • The Memoizer Framework 關于翻譯作者chris,熱衷于java游戲引擎技術,jvm技術,活躍于jxta社區(qū).可以點擊:http://www.matrix.org.cn/user_view.asp?username=chris查看他的個人信息 Java, java, J2SE, j2se, J2EE, j2ee, J2ME, j2me, ejb, ejb3, JBOSS, jboss, spring, hibernate, jdo, struts, webwork, ajax, AJAX, mysql, MySQL, Oracle, Weblogic, Websphere, scjp, scjd
標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
日韩一区三区| 久久精品国产免费| 国产精品极品| 久久精品一本| 国产99久久| 蜜桃免费网站一区二区三区| 亚洲欧美日韩国产综合精品二区 | 99久精品视频在线观看视频| 久久国产电影| 国产精品v一区二区三区| 岛国av在线网站| 欧美亚洲国产精品久久| 亚洲一区黄色| 国产精品一区二区三区www| 狠狠久久伊人| av在线日韩| 蜜臀av亚洲一区中文字幕| 日韩成人午夜精品| 久久伊人亚洲| 国产超碰精品| 亚洲免费播放| 青青伊人久久| 天堂中文av在线资源库| 久久久久伊人| 波多野结衣久久精品| 中文字幕日本一区二区| 7m精品国产导航在线| 精品中文在线| 视频一区中文| 国产精品毛片视频| 色婷婷狠狠五月综合天色拍| 亚洲欧美日本视频在线观看| 日本午夜精品| 久久高清免费| 国产高清日韩| 亚洲成人二区| 日韩午夜视频在线| 欧美91在线| 女同性一区二区三区人了人一| 99视频一区| 免费观看在线综合色| 国产精品亚洲一区二区在线观看| 久久精品国产久精国产| 日韩伦理一区| 亚洲日本在线观看视频| 国产中文欧美日韩在线| 日韩国产综合| 国产精品视区| 欧美日韩亚洲一区三区| 日韩在线看片| 国产精品一区二区美女视频免费看| 欧美成人a交片免费看| 巨乳诱惑日韩免费av| 激情综合婷婷| 国产伊人精品| 精品精品99| 日本欧美在线| 99精品一区| 精品少妇一区| 日韩国产一二三区| 99久久亚洲精品蜜臀| 欧美亚洲福利| 蜜臀av免费一区二区三区| 国产精品va| 老司机久久99久久精品播放免费| 国产成人久久精品一区二区三区| 日本精品在线中文字幕| 国产精品中文字幕制服诱惑| 久久精品国产大片免费观看| 亚洲激情av| 国产一二在线播放| 日韩在线a电影| 欧洲av不卡| 国产精品视频3p| 香蕉视频成人在线观看| 久久久久一区| 88久久精品| 亚洲一区二区三区久久久| 精品午夜久久| 国产日韩视频在线| 亚洲资源av| 国产激情在线播放| 欧美a在线观看| 男女男精品视频网| 国产精品国码视频| 日韩av在线免费观看不卡| 欧美激情另类| 国产精品毛片久久久| 亚洲一区二区成人| 色婷婷综合网| 国产精品九九| 亚洲精选91| 亚洲日韩视频| 欧美成人高清| 一区二区三区四区日本视频| 欧美精品99| 91精品国产自产精品男人的天堂| 水野朝阳av一区二区三区| 黄色成人精品网站| 国产精品久久久久久久久久久久久久久 | 欧美日韩亚洲在线观看| 国产精品99久久免费| 日本aⅴ亚洲精品中文乱码| 日韩专区欧美专区| 久久久久久夜| 精品国产美女a久久9999| 日韩精品社区| 欧美日韩一区二区高清| 亚洲综合图色| 深夜福利亚洲| 日韩精品一区二区三区中文| 亚洲天堂免费| 欧美日本不卡高清| 97久久亚洲| 日韩欧美激情电影| 青青国产精品| 青草综合视频| 麻豆视频久久| 国产日韩1区| 日本亚洲欧美天堂免费| 在线亚洲一区| 亚洲综合激情在线| 91成人精品观看| 日韩美女精品| 精品午夜视频| 久久久久久色| 日韩精品成人| 国产欧美日韩影院| 国产精品观看| 国产精品久久久久久久久妇女| 九九色在线视频| 欧美成人综合| 午夜性色一区二区三区免费视频| 亚洲v天堂v手机在线| 日韩av不卡一区二区| 欧美片第1页综合| 国产一区二区三区成人欧美日韩在线观看| 精品欧美日韩精品| 成人va天堂| 久久午夜视频| 日韩欧美中文字幕电影| 国产乱子精品一区二区在线观看| 国产不卡人人| 久久人人99| 免播放器亚洲| 国产九一精品| 日韩久久一区二区三区| 香蕉国产精品| 91精品一区| 国产精品久久久久久模特 | 人人香蕉久久| 欧美网站在线| 日韩欧美美女在线观看| 成人在线黄色| 91精品蜜臀一区二区三区在线| 亚洲精品乱码久久久久久蜜桃麻豆| 日韩精品国产精品| 久久人人精品| 日韩va亚洲va欧美va久久| 欧美日韩国产一区二区在线观看| 国产一区二区三区探花| 免费黄色成人| 欧美日韩99| 亚洲欧美综合| 日韩1区2区3区| 久久亚洲成人| 欧美精品影院| 激情综合在线| 国产精品男女| 亚洲精品一区二区在线看| 欧美三区不卡| 国产一级一区二区| 国产伦精品一区二区三区千人斩 | 日韩毛片一区| 精品视频黄色| 亚洲免费成人av在线| 日韩大片在线观看| 日韩二区在线观看| 婷婷综合六月| 四虎成人精品一区二区免费网站| 欧美黄色网页| 国产剧情一区二区在线观看| 亚洲福利国产| 久久影院资源站| 日韩三级一区| 蜜桃成人av| 蜜桃精品在线| 卡一卡二国产精品| 在线免费观看亚洲| 中文一区一区三区高中清不卡免费| 四虎精品一区二区免费| 国户精品久久久久久久久久久不卡| 国产精品久久久久久久久免费高清| 午夜久久福利| www.com.cn成人| 91精品视频一区二区| 夜夜精品视频| 亚洲一级黄色| 新版的欧美在线视频| 日韩国产欧美在线播放|