会话跟踪
HTTP是“无状态”协议:客户程序每次读取Web页面,都打开到web服务器的单独的连接,而且,服务器也不自动维护客户的上下文信息。类似客户决定结账时,如何确定之前创建的购物车中哪个属于此客户呢?这种问题有三种解决方案:
cookie、URL重写和隐藏的表单域
cookie处理:
cookie存储购物会话的ID;在后续;连接中,取出当前的会话ID,并使用这个ID从服务器的查找表上提取会话的相关信息。用到两个表:将会话ID与用户关联起来的表和存储用户具体数据的表。
String sessionID = makeUniqueString();
HashMap sessionInfo = new HashMap();
HashMap globalTable = findTableStoringSessions();
globalTable.put(sessionID, sessionInfo);
Cookie sessionCookie = new Cookie("JSESSION", sessionID);
sessionCookie.setPath("/");
response.addCookie(sessionCookie);
这种方式使用cookie是一种绝佳的解决方案,是处理会话常用的方式
url重写:
采用这种方式,客户程序在每个URL尾部加上一些额外数据。这些数据标识当前的会话,服务器将这个标识与它存储的用户相关数据关联起来
隐藏的表单域:
主要缺点是:仅当每个页面都是由表单提交而动态生成才能适用这种方法。
servlet中的会话跟这陪你过
servlet提供一种出色的会话跟踪解决方案:HttpSession API.这个高层接口构筑在cookie和url重写之上
会话跟踪基础:
1、访问与当前请求关联的会话对象
request.getSession获取HttpSession对象。是一个简单的散列表
2、查找与会话相关联的信息
HttpSession的getAttribute
3、存储会话中的信息
setAttribute
4、废弃会话数据
removeAttribute废弃指定的值。invalidate废弃整个会话。logout使客户推出web服务器并作废与用户相关联的所有会话
浏览器会话与服务器会话:
默认,会话跟踪基于存储在浏览器内存中的cookie。
URL重写的话对发往客户的url进行编码:
第一种情况是servlet生成的web页面中含有嵌入的URL。在这种情况下,应该将URL传递给HttpResponse的encodeURL方法。这个方法确定当前是否在使用URL重写,仅在必需时附加会话信息;否则,不做任何更改直接返回传入的URL:
String originalURL = someRelativeOrAbsoluteURL;
String encodeURL = response.encodeURL(originalURL);
out.println("a href=\"" + encodeURL + "\">...</A>")
第二种情况是在sendRedirect调用。这种情况下由于要根据不同的规则确定是否附加会话信息,因此不能使用encodeURL。幸运的是,HttpServletResponse提供encodeRedirectURL方法来处理这种情况:
String originURL = someURL;
String encodedURL = response.encodeRedirectURL(originalURL);
response.sendRedirect(encodedURL);
session:实现购物车的例子:
package com.zhen.test.o5; import com.zhen.domain.Catalog; import com.zhen.domain.CatalogItem; import com.zhen.util.ServletUtilities; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; import java.io.PrintWriter; /** * Created by zhen on 2017-11-15. */ public class CatalogPage extends HttpServlet { private CatalogItem[] items; private String[] itemIDs; private String title; protected void setItems(String[] itemIDs) { this.itemIDs = itemIDs; items = new CatalogItem[itemIDs.length]; for(int i=0; i<items.length; i++){ items[i] = Catalog.getItem(itemIDs[i]); } } protected void setTitle(String title) { this.title = title; } @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { if(items == null){ resp.sendError(HttpServletResponse.SC_NOT_FOUND, "Missing Items."); return; } resp.setContentType("text/html"); PrintWriter out = resp.getWriter(); out.println(ServletUtilities.headWithTitle(title) + "" + "<body bgcolor=\"#fdf5e6\">" + title + "</h1>"); CatalogItem item; for(int i=0; i<items.length; i++){ out.println("<hr>"); item = items[i]; if(item == null){ out.println("<font color=\"red\">Unknow item ID " + itemIDs[i] + "</font>"); }else{ out.println(); String formURL = "com.zhen.test.o5.OrderPage"; formURL = resp.encodeURL(formURL); out.println("<form action=\"" + formURL + "\">" + "<input type=\"hidden\" name=\"itemID\" value=\"" +item.getItemID()+ "\">\n" + "<h2>" + item.getShortDescription() + " ($" + item.getCost() + ")</h2>\n" + item.getLongDescription() + "\n" + "<br>" + "<input type=\"submit\" value=\"Add to Shopping Cart\">\n" + "</form>"); } } out.println("</body></html>"); } } package com.zhen.test.o5; import com.zhen.test.o5.CatalogPage; /** * Created by zhen on 2017-11-15. */ public class KidsBooksPage extends CatalogPage{ public void init(){ String[] ids = {"lewis001", "alexander001", "rowling001"}; setItems(ids); setTitle("All-Time Best Children's Fantasy Books"); } } package com.zhen.test.o5; import com.zhen.domain.ItemOrder; import com.zhen.domain.ShoppingCart; import com.zhen.util.ServletUtilities; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpSession; import java.io.IOException; import java.io.PrintWriter; import java.text.NumberFormat; import java.util.List; /** * Created by zhen on 2017-11-15. */ public class OrderPage extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { HttpSession session = req.getSession(); ShoppingCart cart; synchronized (session) { cart = (ShoppingCart) session.getAttribute("shoppingCart"); if(cart == null){ cart = new ShoppingCart(); session.setAttribute("shoppingCart", cart); } String itemID = req.getParameter("itemID"); if(itemID != null){ String numItemString = req.getParameter("numItems"); if(numItemString == null){ cart.addItem(itemID); }else{ int numItems; try{ numItems = Integer.parseInt(numItemString); }catch (NumberFormatException nef){ numItems = 1; } cart.setNumOrdered(itemID, numItems); } } } resp.setContentType("text/html"); PrintWriter out = resp.getWriter(); String title = "Status of your Order"; out.println(ServletUtilities.headWithTitle(title) + "" + "<body bgcolor=\"#fdf5e6\">\n" + "<h1 align=\"center\">" + title + "</h1>"); synchronized (session){ List itemsOrdered = cart.getItemsOrdered(); if(itemsOrdered.size() == 0){ out.println("<h2><I>No items in your cart...</I></h2"); }else{ out.println("<table border=1 align=\"center\">" + "<tr bgcolor=\"#ffad00\">" + " <th>Item ID</th>" + " <th>Description</th>" + " <th>Unit Cost</th>" + " <th>Number</th>" + " <th>Total Cost</th>" + "</tr>"); ItemOrder order; NumberFormat format = NumberFormat.getCurrencyInstance(); for(int i=0; i<itemsOrdered.size(); i++){ order = (ItemOrder) itemsOrdered.get(i); out.println("<tr>" + " <td>" + order.getItemID() + "</td>" + " <td>" + order.getShortItemDescription() + "</td>" + " <td>" + format.format(order.getUnitCost()) + "</td>" + " <td>" + "<form>" + " <input type=\"hidden\" name=\"itemID\" value=\"" + order.getItemID() + "\"/>\n" + " <input type=\"text\" name=\"numItems\" size=3 value=\"" + order.getNumItems() + "\"/>\n" + " <small><input type=\"submit\" value=\"Update Order\"/></small>" + "</form>" + " </td>" + " <td>" + format.format(order.getTotalCost()) + "</td>" + "</tr>"); } String checkoutURL = resp.encodeURL("/webApp1/html/Checkout.html"); out.println("</table>" + "<form action=\"" + checkoutURL + "\">" + "<big><center>" + "<input type=\"submit\" value=\"Proceed to Checkout\">" + "</center></big></form>"); } out.println("</body></html>"); } } } package com.zhen.test.o5; /** * Created by zhen on 2017-11-15. */ public class TeachBooksPage extends CatalogPage { public void init() { String[] ids = {"hall001", "hall002"}; setItems(ids); setTitle("All-Time Best Computer Books"); } } package com.zhen.domain; /** * @author zhen * Created by zhen on 2017-10-30. */ public class BidInfo { private String itemID = ""; private String itemName = ""; private String bidderName = ""; private String emailAddress = ""; private double bidPrice = 0; private boolean autoIncrement = false; public String getItemID() { return itemID; } public void setItemID(String itemID) { this.itemID = itemID; } public String getItemName() { return itemName; } public void setItemName(String itemName) { this.itemName = itemName; } public String getBidderName() { return bidderName; } public void setBidderName(String bidderName) { this.bidderName = bidderName; } public String getEmailAddress() { return emailAddress; } public void setEmailAddress(String emailAddress) { this.emailAddress = emailAddress; } public double getBidPrice() { return bidPrice; } public void setBidPrice(double bidPrice) { this.bidPrice = bidPrice; } public boolean isAutoIncrement() { return autoIncrement; } public void setAutoIncrement(boolean autoIncrement) { this.autoIncrement = autoIncrement; } public boolean isComplete() { return hasValue(getItemID()) && hasValue(getItemName()) && hasValue(getBidderName()) && hasValue(getEmailAddress()) && (getBidPrice() > 0); } public boolean isPartlyComplete(){ boolean flag = (hasValue(getItemID())) || hasValue(getItemName()) || hasValue(getBidderName()) || hasValue(getEmailAddress()) || (getBidPrice() > 0) || isAutoIncrement(); return flag; } private boolean hasValue(String val) { return (val != null) && (!val.equals("")); } } package com.zhen.domain; /** * Created by zhen on 2017-11-15. */ public class Catalog { private static CatalogItem[] items = { new CatalogItem("hall001", "<I>Core servlets and JavaServer Pages 2nd Edition<I>", "by Marty Hall and Larry Brown", 39.95), new CatalogItem("hall002", "<I>Core web programming , 2nd Edition<I>", "by Marty Hall and Larry Brown", 49.99), new CatalogItem("lewis001", "<I>The Chronicles of Narina<I>", "ny C.S Lewis", 19.95), new CatalogItem("alexander001", "<I>The Prydain Series<I>", "by Lloyed Alexander",19.95), new CatalogItem("rowling001", "The Harry Potter Series", "by J.K. Rowling",59.95) }; public static CatalogItem getItem(String itemID){ CatalogItem item; if(itemID == null){ return null; } for(int i=0; i<items.length; i++){ item = items[i]; if(itemID.equals(item.getItemID())){ return item; } } return null; } } package com.zhen.domain; /** * Created by zhen on 2017-11-15. */ public class CatalogItem { private String itemID; private String shortDescription; private String longDescription; private double cost; public String getItemID() { return itemID; } public void setItemID(String itemID) { this.itemID = itemID; } public String getShortDescription() { return shortDescription; } public void setShortDescription(String shortDescription) { this.shortDescription = shortDescription; } public String getLongDescription() { return longDescription; } public void setLongDescription(String longDescription) { this.longDescription = longDescription; } public double getCost() { return cost; } public void setCost(double cost) { this.cost = cost; } public CatalogItem(String itemID, String shortDescription, String longDescription, double cost) { this.itemID = itemID; this.shortDescription = shortDescription; this.longDescription = longDescription; this.cost = cost; } } package com.zhen.domain; /** * Created by zhen on 2017-11-15. */ public class ItemOrder { private CatalogItem item; private int numItems; public CatalogItem getItem() { return item; } public void setItem(CatalogItem item) { this.item = item; } public int getNumItems() { return numItems; } public void setNumItems(int numItems) { this.numItems = numItems; } public ItemOrder(CatalogItem item, int numItems) { this.item = item; this.numItems = numItems; } public ItemOrder(CatalogItem item) { this.item = item; this.numItems = 1; } public String getItemID(){ return getItem().getItemID(); } public String getShortItemDescription(){ return getItem().getShortDescription(); } public String getLongItemDescription(){ return getItem().getLongDescription(); } public double getUnitCost(){ return getItem().getCost(); } public void incrementNumItems(){ setNumItems(getNumItems() + 1); } public void cancelOrder(){ setNumItems(0); } public double getTotalCost() { return getNumItems() * getUnitCost(); } } package com.zhen.domain; import java.util.ArrayList; import java.util.List; /** * Created by zhen on 2017-11-15. */ public class ShoppingCart { private ArrayList itemsOrdered; public ShoppingCart(){ itemsOrdered = new ArrayList(); } public List getItemsOrdered(){ return itemsOrdered; } public synchronized void addItem(String itemID) { ItemOrder order; for(int i=0; i<itemsOrdered.size(); i++){ order = (ItemOrder) itemsOrdered.get(i); if(order.getItemID().equals(itemID)){ order.incrementNumItems(); return; } } ItemOrder newOrder = new ItemOrder(Catalog.getItem(itemID)); itemsOrdered.add(newOrder); } public synchronized void setNumOrdered(String itemID, int numOrdered) { ItemOrder order; for(int i=0; i<itemsOrdered.size(); i++){ order = (ItemOrder) itemsOrdered.get(i); if(order.getItemID().equals(itemID)){ if(numOrdered <= 0){ itemsOrdered.remove(i); }else{ order.setNumItems(numOrdered); } return; } } ItemOrder newOrder = new ItemOrder(Catalog.getItem(itemID)); itemsOrdered.add(newOrder); } } <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <title>Checking Out</title> </head> <body bgcolor="#FDF5E6"> <h1 align="center">Checking Out</h1> We are sorry, but our electronic credit-card-processing system is currently out of order. Please send a check to: <pre> Marty Hall coreservlets.com, Inc. 6 Meadowsweet Ct., Suite B1 Reisterstown, MD 21136-6020 410-429-5535 hall@coreservlets.com </pre> Since we have not yet calculated shipping charges, please sign the check but not fill in the amount. We will generously do that for you. </body> </html>