java学习笔记—使用HttpSession实现QQ的访问记录(31)
1. 编写QQ空间数据类(QQS.java)
public class QQS {
private static LinkedHashMap<Integer, String> qqs =
new LinkedHashMap<Integer, String>();
static{
qqs.put(, "张三");
qqs.put(, "李四");
qqs.put(, "王五");
qqs.put(, "赵六");
qqs.put(, "田七");
qqs.put(, "焦八");
qqs.put(, "侯九");
qqs.put(, "柳十");
qqs.put(, "小二");
}
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 < ){
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></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);
java学习笔记—使用HttpSession实现QQ的访问记录(31)的更多相关文章
- Java学习笔记4
Java学习笔记4 1. JDK.JRE和JVM分别是什么,区别是什么? 答: ①.JDK 是整个Java的核心,包括了Java运行环境.Java工具和Java基础类库. ②.JRE(Java Run ...
- Java学习笔记之---Servlet
Java学习笔记之---Servlet (一)如何实现Servlet 1.实现javax.servlet.Servlet接口: 2.继承javax.servlet.GenericServlet类: 3 ...
- java学习笔记(1)java的基础介绍 、JDK下载、配置环境变量、运行java程序
java工程师是开发软件的 什么是软件呢? 计算机包括两部分: 硬件: 鼠标.键盘.显示器.主机箱内部的cpu.内存条.硬盘等 软件: 软件包括:系统软件和应用软件 系统软件:直接和硬件交互的软件:w ...
- 0037 Java学习笔记-多线程-同步代码块、同步方法、同步锁
什么是同步 在上一篇0036 Java学习笔记-多线程-创建线程的三种方式示例代码中,实现Runnable创建多条线程,输出中的结果中会有错误,比如一张票卖了两次,有的票没卖的情况,因为线程对象被多条 ...
- 0035 Java学习笔记-注解
什么是注解 注解可以看作类的第6大要素(成员变量.构造器.方法.代码块.内部类) 注解有点像修饰符,可以修饰一些程序要素:类.接口.变量.方法.局部变量等等 注解要和对应的配套工具(APT:Annot ...
- Java学习笔记(04)
Java学习笔记(04) 如有不对或不足的地方,请给出建议,谢谢! 一.对象 面向对象的核心:找合适的对象做合适的事情 面向对象的编程思想:尽可能的用计算机语言来描述现实生活中的事物 面向对象:侧重于 ...
- 0032 Java学习笔记-类加载机制-初步
JVM虚拟机 Java虚拟机有自己完善的硬件架构(处理器.堆栈.寄存器等)和指令系统 Java虚拟机是一种能运行Java bytecode的虚拟机 JVM并非专属于Java语言,只要生成的编译文件能匹 ...
- 0030 Java学习笔记-面向对象-垃圾回收、(强、软、弱、虚)引用
垃圾回收特点 垃圾:程序运行过程中,会为对象.数组等分配内存,运行过程中或结束后,这些对象可能就没用了,没有变量再指向它们,这时候,它们就成了垃圾,等着垃圾回收程序的回收再利用 Java的垃圾回收机制 ...
- 0028 Java学习笔记-面向对象-Lambda表达式
匿名内部类与Lambda表达式示例 下面代码来源于:0027 Java学习笔记-面向对象-(非静态.静态.局部.匿名)内部类 package testpack; public class Test1{ ...
随机推荐
- window 10 下解压缩版MySQL5.7.24的安装
安装步骤: 1.下载mysql-5.6.40-winx64.zip https://cdn.mysql.com//Downloads/MySQL-5.7/mysql-5.7.24-winx64.zip ...
- Android 获取图片转bitmap
1. Resources resources = mContext.getResources(); Drawable drawable = resources.getDrawable(R.drawab ...
- 【转】简述TCP的三次握手过程
TCP握手协议 在TCP/IP协议中,TCP协议提供可靠的连接服务,采用三次握手建立一个连接.第一次握手:建立连接时,客户端发送syn包(syn=j)到服务器,并进入SYN_SEND状态,等待服务器确 ...
- Eclipse安装Freemarker Editor插件
在下面网址里下载freemarker-ide: http://sourceforge.net/projects/freemarker-ide/files/ 下载完成后后解压,由于该IDE里面的free ...
- orchard cms 项目迁移
删除Orchard.Web 下的 App_Data 目录,重新安装项目
- shader一般都是用工具调试的
N卡的话用nvidia的nVidia FX Composer, A卡的话用ATI的render monkey 顶点着色器从何方拿到这些数据?在U3D环境下,答案是从绑定到game object中的Me ...
- C和C++中的不定参数
在初学C的时候,我们都会用到printf函数来写Hello World的程序.在我们看printf函数的声明时,会看到类似于下面代码 int printf(const char * __restric ...
- XSS学习笔记
本片文章是读<<XSS跨站脚本gj剖析与防御>>一书的总结 常见的XSS攻击主要用于1.网络钓鱼,盗用用户账号2.窃取cookies 非httponly情况下,读取docume ...
- linux curl网络库的使用方法
struct V3MemoryStruct { char *memory; size_t size; }; static size_t WriteMemoryCallback(void *conten ...
- CMAKE 教程前两章节学习
原文 https://cmake.org/cmake-tutorial/ 以下是一个循序渐进的教程,它覆盖了CMAKE帮助改进的通常的构建系统的话题.许多话题在<掌握CMAKE>(< ...