明輝手游網(wǎng)中心:是一個(gè)免費(fèi)提供流行視頻軟件教程、在線學(xué)習(xí)分享的學(xué)習(xí)平臺(tái)!

設(shè)計(jì)模式簡(jiǎn)介(二)——Observer

[摘要][目的] 定義對(duì)象間的一種一對(duì)多的依賴(lài)關(guān)系,當(dāng)一個(gè)對(duì)象的狀態(tài)發(fā)生改變時(shí), 所有依賴(lài)于它的對(duì)象都得到通知并被自動(dòng)更新。 [何時(shí)使用?] (1)當(dāng)一個(gè)抽象模型有兩個(gè)方面, 其中一個(gè)方面依賴(lài)于另一方面。將這二者封裝在獨(dú)立的對(duì)象中以使它們可以各自獨(dú)立地改變和復(fù)用。 (2)當(dāng)對(duì)一個(gè)對(duì)象的改變需要同時(shí)改變其它...

[目的]
定義對(duì)象間的一種一對(duì)多的依賴(lài)關(guān)系,當(dāng)一個(gè)對(duì)象的狀態(tài)發(fā)生改變時(shí), 所有依賴(lài)于它的對(duì)象都得到通知并被自動(dòng)更新。

[何時(shí)使用?]
(1)當(dāng)一個(gè)抽象模型有兩個(gè)方面, 其中一個(gè)方面依賴(lài)于另一方面。將這二者封裝在獨(dú)立的對(duì)象中以使它們可以各自獨(dú)立地改變和復(fù)用。
(2)當(dāng)對(duì)一個(gè)對(duì)象的改變需要同時(shí)改變其它對(duì)象, 而不知道具體有多少對(duì)象有待改變。
(3)當(dāng)一個(gè)對(duì)象必須通知其它對(duì)象,而它又不能假定其它對(duì)象是誰(shuí)。換言之, 你不希望這些對(duì)象是緊密耦合的。

[如何使用?]
舉例來(lái)說(shuō):網(wǎng)上商店形式多樣,每個(gè)站點(diǎn)有自己的特色,但也有其一般的共性,單就"商品的變化,以便及時(shí)通知訂戶(hù)"這一點(diǎn),是很多網(wǎng)上商店共有的模式,這一模式類(lèi)似Observerpatern.

具體的說(shuō),如果網(wǎng)上商店中商品在名稱(chēng) 價(jià)格等方面有變化,如果系統(tǒng)能自動(dòng)通知會(huì)員,將是網(wǎng)上商店區(qū)別傳統(tǒng)商店的一大特色.這就需要在商品product中加入Observer這樣角色 ,以便product細(xì)節(jié)發(fā)生變化時(shí),Observer能自動(dòng)觀察到這種變化,并能進(jìn)行及時(shí)的update或notify動(dòng)作.

因此在實(shí)現(xiàn)中必須為要使用Observer的類(lèi)添加如下功能:
(1)Add/Delete observer的功能;
(2)通知(notify或update)所有與之相關(guān)的observer的功能。

示例代碼:(JSP,直接擴(kuò)展Java的API提供的Observer接口Java.util.Observer)

//產(chǎn)品類(lèi) 可供Jsp直接使用UseBean調(diào)用 該類(lèi)主要執(zhí)行產(chǎn)品數(shù)據(jù)庫(kù)插入更新
public class product extends Observable{
  private String name;
  private float price;

  public String getName(){ return name;}
  public void setName(){
   this.name=name;
  //設(shè)置變化點(diǎn)
   setChanged();
   notifyObservers(name);

  }   

  public float getPrice(){ return price;}
  public void setPrice(){
   this.price=price;
  //設(shè)置變化點(diǎn)
   setChanged();
   notifyObservers(new Float(price));

  }

  //以下可以是數(shù)據(jù)庫(kù)更新 插入命令.
  public void saveToDb(){
 


}

我們注意到,在product類(lèi)中 的setXXX方法中,我們?cè)O(shè)置了 notify(通知)方法, 當(dāng)Jsp表單調(diào)用setXXX(如何調(diào)用見(jiàn)我的另外一篇文章),實(shí)際上就觸發(fā)了notisfyObservers方法,這將通知相應(yīng)觀察者應(yīng)該采取行動(dòng)了.

下面看看這些觀察者的代碼,他們究竟采取了什么行動(dòng):

//觀察者NameObserver主要用來(lái)對(duì)產(chǎn)品名稱(chēng)(name)進(jìn)行觀察的
public class NameObserver implements Observer{

  private String name=null;

  public void update(Observable obj,Object arg){

    if (arg instanceof String){

     name=(String)arg;
     //產(chǎn)品名稱(chēng)改變值在name中
     System.out.println("NameObserver :name changet to "+name);

    }


  }

}

//觀察者PriceObserver主要用來(lái)對(duì)產(chǎn)品價(jià)格(price)進(jìn)行觀察的
public class PriceObserver implements Observer{

  private float price=0;

  public void update(Observable obj,Object arg){

    if (arg instanceof Float){

     price=((Float)arg).floatValue();
  
     System.out.println("PriceObserver :price changet to "+price);

    }


  }

}

 

Jsp中我們可以來(lái)正式執(zhí)行這段觀察者程序:

<jsp:useBean id="product" scope="session" class="Product" />
<jsp:setProperty name="product" property="*" />

<jsp:useBean id="nameobs" scope="session" class="NameObserver" />
<jsp:setProperty name="product" property="*" />

<jsp:useBean id="priceobs" scope="session" class="PriceObserver" />
<jsp:setProperty name="product" property="*" />

<%

if (request.getParameter("save")!=null)
{
  product.saveToDb();


  out.println("產(chǎn)品數(shù)據(jù)變動(dòng) 保存! 并已經(jīng)自動(dòng)通知客戶(hù)");

}else{

  //加入觀察者
  product.addObserver(nameobs);

  product.addObserver(priceobs);

%>

  //request.getRequestURI()是產(chǎn)生本jsp的程序名,就是自己調(diào)用自己
  <form action="<%=request.getRequestURI()%>" method=post>

  <input type=hidden name="save" value="1">
  產(chǎn)品名稱(chēng):<input type=text name="name" >
  產(chǎn)品價(jià)格:<input type=text name="price">
  <input type=submit>

  </form>

<%

}

%>

執(zhí)行改Jsp程序,會(huì)出現(xiàn)一個(gè)表單錄入界面, 需要輸入產(chǎn)品名稱(chēng) 產(chǎn)品價(jià)格, 點(diǎn)按Submit后, 還是執(zhí)行該jsp的if (request.getParameter("save")!=null)之間的代碼.


由于這里使用了數(shù)據(jù)javabeans的自動(dòng)賦值概念,實(shí)際程序自動(dòng)執(zhí)行了setName setPrice語(yǔ)句.你會(huì)在服務(wù)器控制臺(tái)中發(fā)現(xiàn)下面信息::

NameObserver :name changet to ?????(Jsp表單中輸入的產(chǎn)品名稱(chēng))PriceObserver :price changet to ???(Jsp表單中輸入的產(chǎn)品價(jià)格); 這說(shuō)明觀察者已經(jīng)在行動(dòng)了.!! 同時(shí)你會(huì)在執(zhí)行jsp的瀏覽器端得到信息:產(chǎn)品數(shù)據(jù)變動(dòng) 保存! 并已經(jīng)自動(dòng)通知客戶(hù)

 

上文由于使用jsp概念,隱含很多自動(dòng)動(dòng)作,現(xiàn)將調(diào)用觀察者的Java代碼寫(xiě)如下:
public class Test {

  public static void main(String args[]){

Product product=new Product();

NameObserver nameobs=new NameObserver();
PriceObserver priceobs=new PriceObserver();

//加入觀察者
product.addObserver(nameobs);
product.addObserver(priceobs);

product.setName("橘子紅了");
product.setPrice(9.22f);

  }

}