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信息。

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

 

此时再访问以上的案例就会导致空指针异常出现。如果需要将网站修复,那么必须使用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);

 

源码面前,了无秘密