JavaBean是一種符合特定規(guī)范的Java對象,在JavaBean中定義了一系列的屬性(也就是成員變量),并提供了訪問和設(shè)置這些屬性的公共方法(也就是getXXX和setXXX方法)。JavaBean可以作為共享數(shù)據(jù)存放在page、request、session和application范圍內(nèi)。在JSP文件中,可以通過專門的標(biāo)簽來定義或訪問JavaBean。例如:<jsp:useBean id="uuwoxin" scopo="page/request/session/application" class="BaiduUser">(--使用BaiduUser類實例化一個對象uuwoxin,相當(dāng)于BaiduUser uuwoxin=new BaiduUser();--) 輸出一個JavaBean的某個屬性到頁面上的時候,可以使用<jsp:getProperty name="uuwoxin" property="password">(--相當(dāng)于uuwoxin.getPassword();--) 設(shè)置一個JavaBean的某個屬性,可以使用<jsp:setProperty name="uuwoxin" property="password" value="uuwoxin_password">(--相當(dāng)于uuwoxin.setPassword("uuwoxin_password");--) 用于實例化JavaBean對象的類是寫在后臺的,比如Tomcat站點中WEB-INF/classes文件夾下,并且需要編譯成字節(jié)碼文件(.class)。 struts框架中的ActionForm Bean就是一種典型的JavaBean。 深入了解JavaBean可以訪問:http://java.sun.com/products/javabeans。 JavaBean的屬性與一般Java程序中所指的屬性,或者說與所有面向?qū)ο蟮某绦蛟O(shè)計語言中對象的屬性是一個概念,在程序中的具體體現(xiàn)就是類中的變量。在JavaBean設(shè)計中,按照屬性的不同作用又細(xì)分為四類:Simple, Index, Bound與Constrained屬性。 3.1.1 Simple屬性 一個簡單屬性表示一個伴隨有一對get/set方法(C語言的過程或函數(shù)在Java程序中稱為"方法")的變量。屬性名與和該屬性相關(guān)的get/set方法名對應(yīng)。例如:如果有setX和getX方法,則暗指有一個名為"X"的屬性。如果有一個方法名為isX,則通常暗指"X"是一個布爾屬性(即X的值為true或false)。例如在下面這個程序中: public class alden1 extends Canvas { string ourString= "Hello"; //屬性名為ourString,類型為字符串 public alden1(){ //alden1()是alden1的構(gòu)造函數(shù),與C++中構(gòu)造函數(shù)的意義相同 setBackground(Color.red); setForeground(Color.blue); } /* "set"屬性*/ public void setString(String newString) { ourString=newString; } /* "get"屬性 */ public String getString() { return ourString; } }
3.1.2 Indexed屬性
一個Indexed屬性表示一個數(shù)組值。使用與該屬性對應(yīng)的set/get方法可取得數(shù)組中的數(shù)值。該屬性也可一次設(shè)置或取得整個數(shù)組的值。例:
public class alden2 extends Canvas { int[] dataSet={1,2,3,4,5,6}; // dataSet是一個indexed屬性 public alden2() { setBackground(Color.red); setForeground(Color.blue); } /* 設(shè)置整個數(shù)組 */ public void setDataSet(int[] x){ dataSet=x; } /* 設(shè)置數(shù)組中的單個元素值 */ public void setDataSet(int index, int x){ dataSet[index]=x; } /* 取得整個數(shù)組值 */ public int[] getDataSet(){ return dataSet; } /* 取得數(shù)組中的指定元素值 */ public int getDataSet(int x){ return dataSet[x]; } }
3.1.3 Bound屬性
一個Bound屬性是指當(dāng)該種屬性的值發(fā)生變化時,要通知其它的對象。每次屬性值改變時,這種屬性就點火一個PropertyChange事件(在Java程序中,事件也是一個對象)。事件中封裝了屬性名、屬性的原值、屬性變化后的新值。這種事件是傳遞到其它的Bean,至于接收事件的Bean應(yīng)做什么動作由其自己定義。
圖3.1是一個簡單Bound屬性示意圖,當(dāng)PushButton的background屬性 與Dialog的background屬性bind時,若PushButton的background屬性發(fā)生變化時,Dialog的background屬性也發(fā)生同樣的變化。 例:
public class alden3 extends Canvas{ String ourString= "Hello"; //ourString是一個bound屬性 private PropertyChangeSupport changes = new PropertyChangeSupport(this); /** 注:Java是純面向?qū)ο蟮恼Z言,如果要使用某種方法則必須指明是要使用哪個對象的方法,在下面的程序中要進(jìn)行點火事件的操作,這種操作所使用的方法是在PropertyChangeSupport類中的。所以上面聲明并實例化了一個changes對象,在下面將使用changes的firePropertyChange方法來點火ourString的屬性改變事件。*/
public void setString(string newString){ String oldString = ourString; ourString = newString; /* ourString的屬性值已發(fā)生變化,于是接著點火屬性改變事件 */ changes.firePropertyChange("ourString",oldString,newString); } public String getString(){ return ourString; } /** 以下代碼是為開發(fā)工具所使用的。我們不能預(yù)知alden3將與哪些其它的Bean組合成為一個應(yīng)用,無法預(yù)知若alden3的ourString屬性發(fā)生變化時有哪些其它的組件與此變化有關(guān),因而alden3這個Bean要預(yù)留出一些接口給開發(fā)工具,開發(fā)工具使用這些接口,把其它的JavaBean對象與alden3掛接。*/
public void addPropertyChangeListener(PropertyChangeLisener l){ changes.addPropertyChangeListener(l); } public void removePropertyChangeListener(PropertyChangeListener l){ changes.removePropertyChangeListener(l); }
通過上面的代碼,開發(fā)工具調(diào)用changes的addPropertyChangeListener方法把其它JavaBean注冊入ourString屬性的監(jiān)聽者隊列l(wèi)中,l是一個Vector數(shù)組,可存儲任何Java對象。開發(fā)工具也可使用changes的removePropertyChangeListener方法,從l中注銷指定的對象,使alden3的ourString屬性的改變不再與這個對象有關(guān)。當(dāng)然,當(dāng)程序員手寫代碼編制程序時,也可直接調(diào)用這兩個方法,把其它Java對象與alden3掛接。 3.1.4 Constrained屬性
一個JavaBean的constrained屬性,是指當(dāng)這個屬性的值要發(fā)生變化時,與這個屬性已建立了某種連接的其它Java對象可否決屬性值的改變。constrained屬性的監(jiān)聽者通過拋出PropertyVetoException來阻止該屬性值 的改變。過程如圖3.2
例:下面程序中的constrained屬性是PriceInCents。
public class JellyBean extends Canvas{ private PropertyChangeSupport changes=new PropertyChangeSupport(this); private VetoableChangeSupport Vetos=new VetoableChangeSupport(this); /*與前述changes相同,可使用VetoableChangeSupport對象的實例Vetos中的方法,在特定條件下來阻止PriceInCents值的改變。*/
...... public void setPriceInCents(int newPriceInCents) throws PropertyVetoException { /* 方法名中throws PropertyVetoException的作用是當(dāng)有其它Java對象否決PriceInCents的改變時,要拋出例外。*/ /* 先保存原來的屬性值*/
int oldPriceInCents=ourPriceInCents; /**點火屬性改變否決事件*/ vetos.fireVetoableChange("priceInCents",new Integer(OldPriceInCents), new Integer(newPriceInCents));
/**若有其它對象否決priceInCents的改變,則程序拋出例外,不再繼續(xù)執(zhí)行下面的兩條語句,方法結(jié)束。若無其它對象否決priceInCents的改變,則在下面的代碼中把ourPriceIncents賦予新值,并點火屬性改變事件*/
ourPriceInCents=newPriceInCents; changes.firePropertyChange("priceInCents", new Integer(oldPriceInCents),new Integer(newPriceInCents)); }
/**與前述changes相同,也要為PriceInCents屬性預(yù)留接口,使其它對象可注冊入PriceInCents否決改變監(jiān)聽者隊列中,或把該對象從中注銷
public void addVetoableChangeListener(VetoableChangeListener l) { vetos.addVetoableChangeListener(l); } public void removeVetoableChangeListener(VetoableChangeListener l){ vetos.removeVetoableChangeListener(l); } ...... } 從上面的例子中可看到,一個constrained屬性有兩種監(jiān)聽者:屬性變化監(jiān)聽者和否決屬性改變的監(jiān)聽者。否決屬性改變的監(jiān)聽者在自己的對象代碼中有相應(yīng)的控制語句,在監(jiān)聽到有constrained屬性要發(fā)生變化時,在控制語句中判斷是否應(yīng)否決這個屬性值的改變。 總之,某個Bean的constrained屬性值可否改變?nèi)Q于其它的Bean或者是Java對象是否允許這種改變。允許與否的條件由其它的Bean或Java對象在自己的類中進(jìn)行定義。
|