設(shè)計模式簡介(二)——Observer
發(fā)表時間:2024-02-06 來源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要][目的] 定義對象間的一種一對多的依賴關(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);
}
}