每次用戶在Eshop.jsp頁內加入一件物品,頁面就向控制servlet發送一個請求。由servlet依次決定適當的動作,然後處理要加入的物品的請求參數。然後它例示一個新的CD Bean(見代碼清單4)表示所選物品,並在會話內更新購物車對象。
代碼清單 4:CD.java
package shopping;
public class CD {
String album;
String artist;
String country;
float price;
int quantity;
public CD() {
album="";
artist="";
country="";
price=0;
quantity=0;
}
public void setAlbum(String title) {
album=title;
}
public String getAlbum() {
return album;
}
public void setArtist(String group) {
artist=group;
}
public String getArtist() {
return artist;
}
public void setCountry(String cty) {
country=cty;
}
public String getCountry() {
return country;
}
public void setPrice(float p) {
price=p;
}
public float getPrice() {
return price;
}
public void setQuantity(int q) {
quantity=q;
}
public int getQuantity() {
return quantity;
}
}
注意:我們在servlet中包括了附加的智能,這樣一來它就能明白,如果一個原先加入過的CD被再次選中,它只需在購物車中為這個CD Bean增加計數就可以了。這個控制servlet也能處理在Cart.jsp中被觸發的動作,比如用戶從購物車中刪除物品或結帳。注意觀察,控制servlet一直在完全掌握著對資源的支配權,它決定在對特定動作的響應中調用哪些資源。例如,購物車狀態的改動,如添加或刪除,會使控制servlet把處理過的請求送至Eshop.jsp頁。這促使該頁重新顯示主視圖,這時購物車中顯示的數據已被更新。如果用戶決定結帳,這個請求在處理後被送至Checkout.jsp頁(見代碼清單5),通過如下所示的調度程序實現:
String url="/jsp/shopping/Checkout.jsp";
ServletContext sc = getServletContext();
RequestDispatcher rd = sc.getRequestDispatcher(url);
rd.forward(req,res);
代碼清單5:Checkout.jsp
<%@ page session="true" import="java.util.*, shopping.CD" %>
<html>
<head>
<title>Music Without Borders Checkout</title>
</head>
<body bgcolor="#33CCFF">
<font face="Times New Roman,Times" size=+3>
Music Without Borders Checkout
</font>
<hr><p>
<center>
<table border="0" cellpadding="0" width="100%" bgcolor="#FFFFFF">
<tr>
<td><b>ALBUM</b></td>
<td><b>ARTIST</b></td>
<td><b>COUNTRY</b></td>
<td><b>PRICE</b></td>
<td><b>QUANTITY</b></td>
<td></td>
</tr>
<%
Vector buylist = (Vector) session.getValue("shopping.shoppingcart");
String amount = (String) request.getAttribute("amount");
for (int i=0; i < buylist.size();i++) {
CD anOrder = (CD) buylist.elementAt(i);
%>
<tr>
<td><b><%= anOrder.getAlbum() %></b></td>
<td><b><%= anOrder.getArtist() %></b></td>
<td><b><%= anOrder.getCountry() %></b></td>
<td><b><%= anOrder.getPrice() %></b></td>
<td><b><%= anOrder.getQuantity() %></b></td>
</tr>
<%
}
session.invalidate();
%>
<tr>
<td> </td>
<td> </td>
<td><b>TOTAL</b></td>
<td><b>$<%= amount %></b></td>
<td> </td>
</tr>
</table>
<p>
<a href="/examples/jsp/shopping/EShop.jsp">Shop some more!</a>
</center>
</body>
</html>
Checkout.jsp僅從會話中取出購物車和所有請求的總數,然後顯示所選的物品及總價格。圖5顯示了結帳時客戶端的視圖。一旦用戶結帳,那麼及時去除會話對象很重要。照顧到這一點,在頁面最後需要有一個session.invalidate()調用。這一處理是必要的,原有有兩個:第一,如果會話不被終止,用戶的購物車就不會重新初始化,當他沒有結帳而試圖開始新一輪購物的時候,他的購物車中仍將保留著他已購買的那些物品。第二,如果用戶未結帳就離開了,那麼這個會話對象不會作廢,仍將占用寶貴的系統資源直到它過期。由於默認的會話有效期是30分鐘,所以在高負荷的系統上,這種情況會使系統資源迅速耗盡。我們當然知道一個應用程序將系統資源耗盡意味著什麼!
圖5:音樂無國界,結帳視圖
圖中文字同圖4。
注意,在這個例子中所有的資源分配都是基於會話的,因為這個模型就是存於會話內的。所以,你必須確保控制SERVLET不被用戶訪問,即使是意外的訪問也不允許。要解決這一問題,可以在控制servlet檢查到一個非法訪問時自動轉向重定向錯誤頁面。(見代碼清單6)
代碼清單 6:error.html
<html>
<body>
<h1>
Sorry, there was an unrecoverable error! <br>
Please try <a href="/examples/jsp/shopping/EShop.jsp">again</a>.
</h1>
</body>
</html>
配置“音樂無國界”
我假定你使用的是Sun公司最新版本的JavaServer Web Development Kit(Java服務器網頁開發工具包-JSWDK)來舉例說明。假設此服務器安裝在jswdk-1.0.1目錄下――這是在Windows中它的默認安裝路徑,“音樂無國界”應用程序的文件應如下配置:
●在 jswdk-1.0.1examplesjsp目錄下建立shopping目錄
●復制Eshop.jsp到jswdk-1.0.1examplesjspshopping
●復制Cart.jsp到jswdk-1.0.1examplesjspshopping
●復制Check.jsp到jswdk-1.0.1examplesjspshopping
●鍵入javac *.java編譯.java文件
●復制 ShoppingServlet.class到 jswdk-1.0.1webpagesWeb-Infservlets
●在jswdk-1.0.1examplesWeb-Infjspbeans目錄下建立shopping目錄
●復制CD.class到jswdk-1.0.1examplesWeb-Infjspbeansshopping
●復制error.html到jswdk-1.0.1webpages
●服務器一旦啟動,你就可以使用http://localhost:8080/examples/jsp/shopping/EShop.jsp訪問這個應用程序
權衡JSP與servlets
在這個例子中,我們仔細考察了JSP Model 2提供的控制水准和靈活性。特別地,我們看到了如何挖掘servlets和JSP的最佳特性,在最大程度上分離內容和表達。正確運用Model2體系結構,可以把所有處理邏輯集中於控制servlet中,讓JSP頁只負責表達或視圖。然而,使用Model 2的弊端是它很復雜。因此,在簡單的應用中Model 1或許更合適。
<全文完>