java学习笔记—使用HttpSession实现QQ的访问记录(31)

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技术。

URLRewritting

分析以上问题的原因:

服务器创建好了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);
原文地址:https://www.cnblogs.com/zhenghongxin/p/4427948.html