1. 编写QQ空间数据类(QQS.java)
public class QQS { private static LinkedHashMap<Integer, String> qqs = new LinkedHashMap<Integer, String>(); static{ qqs.put(10001, "张三"); qqs.put(10002, "李四"); qqs.put(10003, "王五"); qqs.put(10004, "赵六"); qqs.put(10005, "田七"); qqs.put(10006, "焦八"); qqs.put(10007, "侯九"); qqs.put(10008, "柳十"); qqs.put(10009, "小二"); } public static LinkedHashMap<Integer, String> getQqs() { return qqs; } }
2. 编写一个现实QQ数据和浏览记录的页面(ListServlet.java)
public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException { // 获取Session对象 HttpSession session = request.getSession(); // 设置中文数据 response.setCharacterEncoding("utf-8"); response.setContentType("text/html;charset=utf-8"); // 获取输出流 PrintWriter out = response.getWriter(); // 获取QQS数据 LinkedHashMap<Integer, String> qqs = QQS.getQqs(); Set<Map.Entry<Integer, String>> set = qqs.entrySet(); Iterator<Map.Entry<Integer, String>> it = set.iterator(); // 输出页面结构 out.println("<html><head><title>QQ列表</title><style>a{margin-right:20px;}</style></head><body>"); out.println("<hr/><br/>"); out.println("<h3>QQ列表</h3>"); out.println("<hr/><br/>"); // 循环输出QQ空间的超链接 while(it.hasNext()){ Map.Entry<Integer, String> entry = it.next(); Integer num = entry.getKey(); String name = entry.getValue(); out.println("<a href=\"/day08/store?num="+num+"\">"+name+"</a>"); } // 输出浏览的记录信息 out.println("<hr/><br/>"); out.println("<h3>QQ浏览记录</h3>"); out.println("<hr/><br/>"); // 获取访问记录数据 String history = (String) session.getAttribute("history"); if(history == null){ out.println("<font color=\"red\">对不起,目前没有访问记录...</font>"); }else{ // 循环遍历用户访问的记录数据 String[] nums = history.split(","); for(String num:nums){ String name = QQS.getQqs().get(Integer.parseInt(num)); out.println(name+" ,"); } } // 关闭页面结构 out.println("</body></html>"); }
3. 编写一个存储浏览QQ空间的页面(StoreQQServlet.java)
public void doGet(HttpServletRequest request, HttpServletResponse response)throws ServletException, IOException { // 获取Session对象 HttpSession session = request.getSession(false); // 获取请求参数 String num = request.getParameter("num"); // 获取Session中的数据 String history = (String) session.getAttribute("history"); // 判断数据 if(history == null){ // 第一次访问 session.setAttribute("history", num); // history=10001 }else{ // 访问多次 session.setAttribute("history", history+","+num); // 设置num的数量和显示的顺序 String[] qqs = history.split(","); // 将数组转换为方便操作的集合 List<String> list = Arrays.asList(qqs); // 将List转换为LinkedList便于操作数据 LinkedList<String> linked_list = new LinkedList<String>(); linked_list.addAll(list); // 判断出现的QQ次数 if(qqs.length < 3 ){ if(linked_list.contains(num)){ // history=10002,1003 // 如果包含 linked_list.remove(num); linked_list.addFirst(num); }else{ // history=1004,10002,1003 // 不包含 linked_list.addFirst(num); } }else{ // >= 3 if(linked_list.contains(num)){ // history=10002,10003,10004 10004 // 如果包含 linked_list.remove(num); linked_list.addFirst(num); }else{ // history= 10005 ,10002,10003 // 不包含 linked_list.removeLast(); linked_list.addFirst(num); } } // 次数好了,顺序好了的访问记录linked_list StringBuffer sb = new StringBuffer(); for(String new_num:linked_list){ sb.append(new_num+","); } String new_history = sb.toString(); session.setAttribute("history", new_history); } // 重定向到QQ列表页面 response.sendRedirect("/day08/list"); }
以上的代码将用户的浏览记录存储在了session对象中,但是该对象是在服务器内存中的,且有有效的时间限制,如果时间到了,那么session就会被销毁。
默认的时间为半个小时(30分钟)。
4 配置Session的有效时间
在每一个网站的web.xml中可以配置该网站创建的session对象的有效时间。注意的是配置时单位是分钟。
Thread.slessp(毫秒单位)、Cookie.setMaxAge(秒单位)、session(分钟单位)
<session-config> <session-timeout>2</session-timeout> 单位是分钟 </session-config>
5 Cookie的禁用
Cookie可以利用客户端存储会话数据。
HttpSession可以利用Cookie存储SessionID信息。
其实在浏览器的设置中可以拒绝网站发送回来的Cookie信息。
此时再访问以上的案例就会导致空指针异常出现。如果需要将网站修复,那么必须使用URLRewriting技术。
分析以上问题的原因:
服务器创建好了Session对象,但是由于浏览器禁止了Cookie的接收,那么服务器无法将创建好的Session的ID值以Set-Cookie的响应头方式发送给浏览器进行存储,那么在第二访问的时候也就不会懈怠SessionID,因此无法找到Session。
常用的方法
String encodeRedirectURL(String url) 给指定的重定向路径后添加Sessionid信息
String encodeURL(String url) 给普通的URL地址添加Sessionid信息
实施的原则:
“将页面中的所有的URL地址全部使用以上方法进行重新编码”
1 修改以上程序
1 ListServlet.java
String path = "/day08/store?num="+num; path = response.encodeURL(path); out.println("<a href='"+path+"'>"+name+"</a>");
2. StoreQQServlet.java
String path = "/day08/list"; path = response.encodeRedirectURL(path); response.sendRedirect(path);