設(shè)計(jì)模式簡(jiǎn)介(二)——Observer
發(fā)表時(shí)間:2024-06-05 來(lái)源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要][目的] 定義對(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);
}
}