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

設(shè)計模式簡介(二)——Observer

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

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

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

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

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

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

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

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

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

  }   

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

  }

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


}

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

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

//觀察者NameObserver主要用來對產(chǎn)品名稱(name)進行觀察的
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)品名稱改變值在name中
     System.out.println("NameObserver :name changet to "+name);

    }


  }

}

//觀察者PriceObserver主要用來對產(chǎn)品價格(price)進行觀察的
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中我們可以來正式執(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ù)變動 保存! 并已經(jīng)自動通知客戶");

}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)品名稱:<input type=text name="name" >
  產(chǎn)品價格:<input type=text name="price">
  <input type=submit>

  </form>

<%

}

%>

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


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

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

 

上文由于使用jsp概念,隱含很多自動動作,現(xiàn)將調(diào)用觀察者的Java代碼寫如下:
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);

  }

}