要了解Session的底层工作原理。我们还是先看在一个会话过程中,同一个浏览器在访问多个web资源的情况好了,大致分为以下几个步骤:
1,浏览器访问某个Servlet,这时如果服务器要从请求对象中获取Session对象(第一次获取也是创建),那么服务器会为这个Session对象创建一个id:JSESSIONID
2,同时在对浏览器的响应过程中,这个Session会将JSESSIONID这个id以COOKIE形式回送给客户端浏览器,记住,这时候COOKIE服务器没有设置有效时间,因此是存在浏览器的缓存中,而不是在硬盘文件。
3,当用户继续在这个会话过程中访问其他Servlet,这时候这个Servlet再从请求对象中获取Session对象,注意这时候获取Session对象是从浏览器发来的请求中查询是否有名为JSESSIONID的这个COOKIE,如果有,那么这个Session就不用再创建,而是直接根据查询服务器中这个相同JSESSIONID值的Session,换句话说就可以取得之前存在这个Session中的数据。
总结来说,Session是基于COOKIE的。
(注:COOKIE并不是万能的,Session首先是依据COOKIE,但是有时候COOKIE不能用,这时候Session会查询发来请求的URL地址是否有JSESSIONID。)
Session的隐藏COOKIE,我们可以做个小实验来验证下,在【myservlet】这个web工程下创建两个Servlet,分别命名为SessionDemo1和SessionDemo2:
在SessionDemo1代码为:
HttpSession session = request.getSession(); String data = "Message from SessionDemo"; session.setAttribute("data", data);
在SessionDemo2代码为:
HttpSession session = request.getSession(); System.out.println((String)session.getAttribute("data"));
我们在浏览器中打开HttpWatch,来访问SessionDemo1,因为是首次访问Servlet,查看SessionDemo1给浏览器的响应:
确确实实服务器发送回浏览器有这个JSESSIONID名称的COOKIE,这时候如果我们再在打开的浏览器去访问SessionDemo2,那么在HttpWatch中观察请求包的内容发现:
再次访问服务器时,浏览器就会带着这个名为JSESSIONID的COOKIE给服务器,服务器正是通过这个COOKIE中的JSESSIONID值去服务器中查找之前为该浏览器创建的Session。
如果我们将浏览器关闭,由于这个COOKIE没有设置“setMaxAge”,因此这个COOKIE只存在于浏览器的缓冲,浏览器关闭即被销毁。如果想使关闭浏览器之后,Session还能存在,我们就要人为的覆盖这个Session的COOKIE,并设置覆盖COOKIE的有效时间和有效路径。而这个COOKIE的值,也就是JSESSIONID的值,可以通过Session的getId()方法得到。
1,覆盖有效时间:
注意,服务器在为浏览器创建Session后,在用户没有操作的情况下(或者浏览器关闭后)默认为其维护30分钟。这点可以从Tomcat的【web.xml】文件中可以看出:
当然我们从这里也可以修改服务器默认的销毁无操作的Session时间。
当然如果我们不要全局设置所有服务器中Session的销毁时间,就在每个web应用中的web.xml文件中自定义添加
注:我们还可以通过Session对象的invalidate()方法,将某个Session进行立刻销毁。
对此,如果我们要覆盖一个Session的COOKIE并保存在硬盘文件中,我们设置的COOKIE有效时间就不要超过服务器默认的session-timeout时间。
2,覆盖有效路径:
如果我们创建一个COOKIE对象,没有设置“setPath”,那么COOKIE的有效路径为创建该COOKIE的程序(通常为某个Servlet),即只有访问了这个程序时浏览器才会带着COOKIE过去,那实在是“人脉不通”,访问这个web应用的其他资源就无法再使用Session了。
我们看看刚才的第一次访问Servlet时,服务器为浏览器创建的Session中的COOKIE的有效路径:
可以看到这个服务器默认将JSESSIONID这个COOKIE的有效路径设置为创建这个Session的web工程根目录。所以我们要覆盖Session中的COOKIE时也应该设置路径为该web工程根目录。
好,接下来对上面那个Servlet的例子进行改造,我们只需要在SessionDemo1中修改就行,因为这个首次将Session的COOKIE返回给客户端,修改后代码如下:
HttpSession session = request.getSession(); String data = "Message from SessionDemo"; session.setAttribute("data", data); COOKIE COOKIE = new COOKIE("JSESSIONID", session.getId()); COOKIE.setMaxAge(30*60); COOKIE.setPath("/myservlet"); response.addCOOKIE(COOKIE);
这样,当我们打开浏览器访问了SessionDemo1之后,就能在存放COOKIE的目录中找到该COOKIE,如果我们通过HttpWatch来查看可以看到重名的这个COOKIE:
虽然JSEESIONID这个COOKIE重名了,没有关系,因为其值都是一样的,并且如果我们将浏览器关闭后,没有设置COOKIE有效时间的(也是原先Session发来的)COOKIE将不复存在(存在浏览器缓存中,浏览器关闭就被销毁),这时重新打开一个浏览器,再去访问SessionDemo2依然能获取到原来Session中保存的内容:
注意,这是另外打开浏览器窗口访问的SessionDemo2!!另附:
通过这里我们可以看到,我们人为地将原先Session定义的COOKIE给替换了,而Session并不知道,只要能获得“JSESSIONID”这个COOKIE,它就认为COOKIE是存在的,可以从这个COOKIE中id值获取以前保存的信息,因此我们实现了一台主机共享一个Session,此时,当浏览器关闭,或者说结束一个会话后,依然能获取Session来获取之前保存的数据。