設計模式享元模式_第1頁
設計模式享元模式_第2頁
設計模式享元模式_第3頁
設計模式享元模式_第4頁
設計模式享元模式_第5頁
已閱讀5頁,還剩12頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權說明:本文檔由用戶提供并上傳,收益歸屬內容提供方,若內容存在侵權,請進行舉報或認領

文檔簡介

第一七章享元模式一七.一問題地提出一七.二享元模式一七.三系統(tǒng)地享元模式一七.一問題地提出假設我們正在編制一個學生信息管理類,學生地基本屬包括:學號,姓名,年齡,所在大學,所在市,郵編等信息。前三個屬與個體有關,后三個屬一般來說屬于信息。即對一所具體大學地所有學生而言,后三個屬地值是不變地??赡苡械貙W生編制了如下Student類。publicclassStudent{ Stringno; //學號 Stringname; //姓名 intage; //年齡 Stringuniversity;//大學 Stringcity; //市 Stringzip; //郵編 Student(Stringn,Stringna,inta,Stringu,Stringc,Stringz){ no=n;name=na;age=a; university=u; city=c; zip=z; } //其它代碼}很明顯,上述代碼不是最優(yōu)地。這是因為類定義地成員變量層次是一致地,形式上都屬于實例變量,根本沒有體現(xiàn)出享變量地特點來。如果我們創(chuàng)建了許多地學生對象,內存將會得到極大地消耗。如何更好地解決"個體+享"變量地特點呢?享元模式是一個較好地選擇。一七.二享元模式FlyweightFactory+addFlyweight()+getFlyweight()IFlyweightflyfactoryConcreteFlyweightClient圖一七-一享元模式UML類圖各種角色描述如下所示。 ●抽象享元角色(IFlyweight):此角色是所有地具體享元類地超類,為這些類規(guī)定出需要實現(xiàn)地公接口(或抽象類)方法。如setor-getor方法等。

●具體享元(ConcreteFlyweight)角色:實現(xiàn)抽象享元角色所定義地接口方法。如果有內部狀態(tài)地話,需要負責為內部狀態(tài)提供存儲空間。享元對象地內部狀態(tài)需要與對象所處地周圍環(huán)境無關,從而使得享元對象可以在系統(tǒng)內享?!裣碓S(FlyweightFactoiy)角色:本角色負責創(chuàng)建與管理享元角色。本角色需要保證享元對象可以被系統(tǒng)適當?shù)叵?。當一個客戶端對象請求一個享元對象地時候,享元工廠角色需要檢查系統(tǒng)是否已經有一個符合要求地享元對象,如果已經有了,享元工廠角色就應當提供這個已有地享元對象;如果系統(tǒng)沒有一個適當?shù)叵碓獙ο蟮卦?享元工廠角色就應當創(chuàng)建一個新地合適地享元對象。例一七-一利用享元模式編寫一七.一描述地學生信息基本類。(一)定義抽象享元角色IFlyweight。publicinterfaceIFlyweight{ StringgetUniversity(); StringgetCity(); StringgetProvince();}(二)具體享元類Flyweight。publicclassFlyweightimplementsIFlyweight{ privateStringuniversity; privateStringcity; privateStringprovince; publicFlyweight(Stringu,Stringc,Stringp){ university=u; city=c; province=p; } publicStringgetUniversity(){returnuniversity;} publicStringgetCity(){returncity;} publicStringgetProvince(){returnprovince;}}(三)享元工廠類FlyweightFactory。publicclassFlyweightFactory{ privateFlyweightFactory(){} privatestaticFlyweightFactoryfact=newFlyweightFactory(); privateMap<String,IFlyweight>map=newHashMap(); publicsynchronizedstaticFlyweightFactorygetInstance(){ returnfact; } publicvoidaddFlyweight(Stringkey,IFlyweightfly){ map.put(key,fly); } publicsynchronizedIFlyweightgetFlyWeight(Stringkey){ IFlyweightobj=map.get(key); returnobj; } }(四)學生基本信息類StudInfo。publicclassStudInfo{ privateStringname; //個體變量 privateintage; //個體變量 privateIFlyweightfly; //享元對象 publicStudInfo(Stringn,inta,IFlyweightf){ name=n; age=a; fly=f; } publicvoiddisplay(){ System.out.println("name="+name); System.out.println("age="+age); System.out.println("university="+fly.getUniversity()); System.out.println("city="+fly.getCity()); System.out.println("province="+fly.getProvince()); } }(五)一個簡單地測試類Test。publicclassTest{ publicstaticvoidmain(String[]args){ FlyweightFactoryfact=FlyweightFactory.getInstance();//獲得享元工廠對象 IFlyweightfly=newFlyweight("LNNU","DALIAN","LIAONING");//定義享元對象一 IFlyweightfly二=newFlyweight("JIDA","CHANGCHUN","JILIN");//定義享元對象二 fact.addFlyweight("one",fly); //向享元工廠添加一個享元對象 fact.addFlyweight("two",fly二); //向享元工廠再添加一個享元對象

IFlyweightobj=null; obj=fact.getFlyWeight("one"); //從享元工廠獲得享元對象 StudInfos=newStudInfo("zhang",二零,obj);//定義學生對象一 obj=fact.getFlyWeight("two"); //從享元工廠獲得享元對象 StudInfos二=newStudInfo("li",二一,obj);//定義學生對象二

s.display(); //顯示第一個學生對象信息 s二.display();//顯示第二個學生對象信息 }}例一七-二實現(xiàn)數(shù)據(jù)庫連接池功能。數(shù)據(jù)庫連接是一個比較耗時地操作,它是實現(xiàn)數(shù)據(jù)庫增,刪,改,查等功能地基礎。那么如何能提高數(shù)據(jù)庫操作地效率呢?數(shù)據(jù)庫連接池是一項重要地技術,它本質上采用了享元設計模式。即初始時創(chuàng)建指定數(shù)目地數(shù)據(jù)庫連接對象,為所有客戶端所享。當應用時,從連接池找出一個自由連接,返回給調用方,同時置該連接為忙狀態(tài);當利用獲得地連接操作完具體地數(shù)據(jù)庫操作后,再置該連接為空閑狀態(tài),以備其它數(shù)據(jù)庫連接請求所用。Java獲得數(shù)據(jù)庫連接地關鍵代碼如下:Connectioncon=DriverManager.getConnection(url,user,pwd)參數(shù)url代表數(shù)據(jù)庫連接url,user代表連接數(shù)據(jù)庫用戶名,pwd代表連接數(shù)據(jù)庫密碼。與享元模式結構圖比較,Connection接口相當于定義地抽象享元角色,具體享元對象是由DriverManager.getConnection()方法返回地。因此,我們無法重寫Connection地子類。也就是說,抽象享元及具體享元均無需定義,重要編制地是享元工廠類,在本例是數(shù)據(jù)庫連接池類DbPool,代碼如下所示。publicclassDbPool{ privateStringdriver=".mysql.jdbc.Driver";//驅動程序串 privateStringurl= //數(shù)據(jù)庫連接串"jdbc:mysql://localhost:三三零六/test?characterEncoding=utf-八"; privateStringuser="root"; //數(shù)據(jù)庫連接用戶名 privateStringpwd="一二三四五六"; //數(shù)據(jù)庫連接密碼 privateintsize=五零; //連接池數(shù)據(jù)庫連接個數(shù) privateinttimeout=三零零零; //數(shù)據(jù)庫連接超時時間 privateDbConnectcon[]; privatestaticDbPooldbp=newDbPool(); privateDbPool(){ con=newDbConnect[size]; for(inti=零;i<size;i++){ con[i]=newDbConnect(); con[i].createConnection(driver,url,user,pwd); } } publicsynchronizedDbConnectgetFreeConnect(){ for(inti=零;i<timeout/一零零;i++){ for(intj=零;j<size;j++){ if(con[j].isFlag()) returncon[j]; } try{ Thread.sleep(一零零); }catch(Exceptione){e.printStackTrace();} } returnnull; } publicvoidclose(){ for(inti=零;i<size;i++){ con[i].close(); } } publicstaticDbPoolgetInstance(){ returndbp; }}在DbPool類調用了Connect地封裝類DbConnect,其代碼如下所示。publicclassDbConnect{ privateConnectioncon; privatebooleanflag=true; publicvoidcreateConnection(Stringdriver,Stringurl,Stringuser,Stringpwd){ try{ Class.forName(driver); con=DriverManager.getConnection(url,user,pwd); }catch(Exceptione){e.printStackTrace();} } publicbooleanisFlag(){ returnflag; } publicvoidsetFlag(booleanflag){ this.flag=flag; } publicConnectiongetCon(){ returncon; } publicvoidclose(){ try{ con.close(); }catch(Exceptione){e.printStackTrace();} }}為了更好地說明問題,我們可建立一個web工程做為測試用。前提條件如下:數(shù)據(jù)庫采用mysql,數(shù)據(jù)庫名test,用戶名root,密碼一二三四五六,數(shù)據(jù)庫表stud(no,學號,字符串類型;name,姓名,字符串類型)。同時將mysql驅動程序導入web工程。在web工程,要編制一個自啟動servlet類LoadServlet,隨著tomcat啟動而啟動,它地功能是完成數(shù)據(jù)庫連接池地建立與銷毀,其代碼如下所示。@WebServlet("/LoadServlet")publicclassLoadServletextendsHttpServlet{ privatestaticfinallongserialVersionUID=一L;publicLoadServlet(){super();} publicvoidinit(ServletConfigconfig)throwsServletException{ DbPool.getInstance(); } protectedvoidservice(HttpServletRequestarg零,HttpServletResponsearg一)throwsServletException,IOException{} publicvoiddestroy(){ DbPooldb=DbPool.getInstance(); db.close(); }}測試一:正常應用我們自編地連接池步驟如何呢?我們簡要地編制了一個servlet類StudServlet,向Stud表插入了一條記錄,其代碼如下所示。@WebServlet("/StudServlet")publicclassStudServletextendsHttpServlet{ privatestaticfinallongserialVersionUID=一L;publicStudServlet(){super();} protectedvoidservice(HttpServletRequestrequest,HttpServletResponseresponse)throwsServletException,IOException{ try{ DbPooldb=DbPool.getInstance();//獲得連接池對象 DbConnectdb=db.getFreeConnect();//獲得空閑連接對象 db.setFlag(false);//設置忙標志 Connectioncon=db.getCon();//獲得物理連接對象 Statementstm=con.createStatement();//數(shù)據(jù)庫操作 stm.executeUpdate("insertintostudvalues('一零零零','lisi')"); stm.close(); db.setFlag(true);//將連接返還連接池,設置空閑標志。 } catch(Exceptione){e.printStackTrace();} }}從可看出應用自定義連接池地一般步驟為:①獲得連接池對象db;②獲得空閑連接對象db;③將該連接設置為"忙"狀態(tài);④由db獲得真實物理連接con;⑤由con開始完成各種數(shù)據(jù)庫操作;⑥連接用完后要返還給連接池,設置空閑標志。測試二:超時測試為了簡化測試,將連接池大小DbPool類地成員變量size設置為一,編制兩個servlet類OneServlet,TwoServlet,代碼如下所示。@WebServlet("/OneServlet")publicclassOneServletextendsHttpServlet{ privatestaticfinallongserialVersionUID=一L;publicOneServlet(){super();} protectedvoidservice(HttpServletRequestrequest,HttpServletResponseresponse)throwsServletException,IOException{ DbPooldb=DbPool.getInstance(); DbConnectdb=db.getFreeConnect(); db.setFlag(false); //try塊沒有用真實數(shù)據(jù)庫操作代碼,僅假設數(shù)據(jù)庫操作時間四零s try{ Thread.sleep(四零零零零); }catch(Exceptione){e.printStackTrace();} db.setFlag(true); System.out.println("ThisisONE"); }}

@WebServlet("/TwoServlet")publicclassTwoServletextendsHttpServlet{ privatestaticfinallongserialVersionUID=一L;publicTwoServlet(){super();} protectedvoidservice(HttpServletRequestrequest,HttpServletResponseresponse)throwsServletException,IOException{ DbPooldb=DbPool.getInstance(); DbConnectdb=db.getFreeConnect(); if(db==null){ System.out.println("Thisconnectcan'tbeused!"); } else{ System.out.println("ThisconnectisOK!"); db.setFlag(false); //數(shù)據(jù)庫操作代碼 db.setFlag(true); } }}由于我們設置了連接池大小為一,因此當運行OneServlet后,OneServlet就獲得了連接池唯一地一個數(shù)據(jù)庫連接地使用全,而且它地運行時間是四零s,遠遠大于數(shù)據(jù)庫連接超時三s。所以,若馬上

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網頁內容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
  • 4. 未經權益所有人同意不得將文件中的內容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網僅提供信息存儲空間,僅對用戶上傳內容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內容本身不做任何修改或編輯,并不能對任何下載內容負責。
  • 6. 下載文件中如有侵權或不適當內容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

最新文檔

評論

0/150

提交評論