我们要认识自己,不断审视自己,这是我们一生中要努力追找地目标。哪一种事情适合自己干?应该做什么,不应该做什么?有了一个坚定的目标,并付诸行动,成功指日可待。
每次用户在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 rel="nofollow noopener noreferrer" 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 rel="nofollow noopener noreferrer" 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或许更合适。 <全文完> 本文实例讲解JSP Model2体系结构(下)到此结束。对推销员而言,售后服务不是随便可以做好的,仍有很重的分量。小编再次感谢大家对我们的支持!