jsessionid
在服务器端,我们用惯了session.setAttribute("",userInfo)这样的一行代码,估计你很少想到:服务器与浏览器之间是如何保持会话状态的。好了,先引用一些文章的精彩片段:http://www.xxx.com/xxx_app;jsessionid=xxxxxxxxxx?a=x&b=x。
这跟一般的url基本一样,只有一个地方有区别,那就是“;jessionid=xxxxxxxx”。这个参数有时候有,有时候又没有,说它是参数可又跟一般传递的参数不同,它是紧跟在url后面用分号来分隔的 ,用一般的request.getParameter()方法还取不到jsessionid 。
session的实现方式
做web开发的同学都知道,http是无状态的会话协议,也就是说无法保存用户的信息。那如果有一些信息需要在用户的浏览活动中一直保持,该怎么做呢?我们可以把这些信息在每次请求的时候作为参数传递给服务器,但这样做既麻烦又耗费资源,这时候就体现出了session的重要性。session是web开发中不可或缺的一个特性。它是对于一个特定的用户请求,在web服务器上保存的一个全局变量。有了它我们就可以把用户的一些信息保存在服务器上,而不用在服务器和客户端之间来回传递。知道了session的作用,那session是怎么实现的呢?服务器上为每个用户都保存了一个session,那当用户请求过来的时候是怎么知道某一个用户应该对应哪个session呢?这时jsessionid就派上用场了。每一个session都有一个id来作为标识,这个id会传到客户端,每次客户端请求都会把这个id传到服务器,服务器根据id来匹配这次请求应该使用哪个session。jsessionid就是客户端用来保存sessionid的变量,主要是针对j2ee实现的web容器,没有研究过其他语言是用什么变量来保存的。一般对于web应用来说,客户端变量都会保存在cookie中,jsessionid也不例外。不过与一般的cookie变量不同,jsessionid是保存在内存cookie中的,在一般的cookie文件中是看不到它的影子的。内存cookie在打开一个浏览器窗口的时候会创建,在关闭这个浏览器窗口的时候也同时销毁。这也就解释了为什么session变量不能跨窗口使用,要跨窗口使用就需要手动把jsessionid保存到cookie里面。
jsessionid的作用
在以上的文字中我们了解了session的实现原理,同时也知道了session跟jsessionid紧密不可分割的联系。只有通过jsessionid才能使session机制起作用,而jsessionid又是通过cookie来保存。看到这里,也许你会发现一个问题,如果用户禁用了cookie,那jsessionid不是就不能保存了吗?session不是不起作用了吗?我们真的对此束手无策了吗?当然不是。在用户禁用了cookie时候,我们可以通过url重写来实现jsessionid的传递。这就是我上面指出的那样的url:http://www.xxx.com/xxx_app;jsessionid=xxxxxxxxxx?a=x&b=x..。jessionid通过这样的方式来从客户端传递到服务器端,从而来标识session。注意一点,jsessionid跟一般的url参数传递方式是不同的,不是作为参数跟在?后面,而是紧跟在url后面用;来分隔。这样在用户禁用cookie的时候我们也可以传递jsessionid来使用session了,只不过需要每次都把jseesionid作为参数跟在url后面传递。那这样岂不是很麻烦,每次请求一个url都要判断cookie是否可用,如果禁用了cookie,还要从url里解析出jsessionid,然后跟在处理完后转到的url后面,以保持jsessionid的传递。这些问题sun当然已经帮我们想到了,所以提供了2个方法来使事情变得简单:response.encodeURL()和response.encodeRedirectURL()。这2个方法会判断cookie是否可用,如果禁用了会解析出url中的jsessionid,并连接到指定的url后面,如果没有找到jessionid会自动帮我们生成一个。至于为什么要有2个方法?这2个方法有什么不同?google了一下,说是这2个方法在判断是否要包含jsessionid的逻辑上会稍有不同。在调用HttpServletResponse.sendRedirect前,应该先调用encodeRedirectURL()方法,否则可能会丢失Sesssion信息。这2个方法的使用方法如:response.sendRedirect(response.encodeURL("/myapp/input.jsp"));。如果cookie没有禁用,我们在浏览器地址栏中看到的地址是这样的:/myapp/input.jsp,如果禁用了cookie,我们会看到:/myapp/input.jsp;jsessionid=73E6B2470C91A433A6698C7681FD44F4。所以,我们在写web应用的时候,为了保险起见,应该在程序里的每一个跳转url上都使用这2个方法,来保证session的可用性。
启动你的tomcat,打开FireFox(爱得不得了,一定要安装FireBug),输入localhost就行,打开firebug,点网络,你会看到,浏览器与服务器会话的信息,给出浏览器
(1)第一次请求服务器:
浏览器的请求头信息
Host |
localhost
|
User-Agent |
Mozilla/5.0 (Windows; U; Windows NT 5.1; zh-CN; rv:1.9.2.6) Gecko/20100625 Firefox/3.6.6
|
Accept |
text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
|
Accept-Language |
zh-cn,zh;q=0.5
|
Accept-Encoding |
gzip,deflate
|
Accept-Charset |
GB2312,utf-8;q=0.7,*;q=0.7
|
Keep-Alive |
115
|
Connection |
|
服务器响应头信息
Server |
Apache-Coyote/1.1
|
Set-Cookie |
JSESSIONID=64D21B4D69DFB3041B6375C1932BD6CB; Path=/
|
Content-Type |
text/html;charset=UTF-8
|
Content-Language |
zh-CN
|
Content-Length |
242
|
Date |
Mon, 28 Jun 2010 02:35:29 GMT
|
(2)第二次请求服务器:
浏览器的请求头信息
Host |
localhost
|
User-Agent |
Mozilla/5.0 (Windows; U; Windows NT 5.1; zh-CN; rv:1.9.2.6) Gecko/20100625 Firefox/3.6.6
|
Accept |
text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
|
Accept-Language |
zh-cn,zh;q=0.5
|
Accept-Encoding |
gzip,deflate
|
Accept-Charset |
GB2312,utf-8;q=0.7,*;q=0.7
|
Keep-Alive |
115
|
Connection |
keep-alive
|
Cookie |
JSESSIONID=64D21B4D69DFB3041B6375C1932BD6CB
|
服务器响应头信息
Server |
Apache-Coyote/1.1
|
Content-Type |
text/html;charset=UTF-8
|
Content-Language |
zh-CN
|
Content-Length |
242
|
Date |
Mon, 28 Jun 2010 02:37:51 GMT
|
重复第三次,每四次...第N次请求服务器,浏览器和服务器的请求头信息都是与第二次请求服务器是一样的。
(3)但是,如果你在服务器端加入如下一行代码:
Log.info("SessionId:" + request.getSession().getId());
你会看到,当你第一次请求服务器时,就会默认有一个新的session被创建,而且在session的有效时间范围内,这个输出值是不会变的,否则,服务器会重新创建一个session,自然,sessionId也就不同了,这段代码的输出自然也会不同了。
(4)你必须注意这一点:你用的是浏览器与服务器通信:
有一些事情是浏览器帮助我们去做了,那就是:当你第一次与服务器通信时,浏览器会保存服务器返回的 Set-Cookie 这个健的值( JSESSIONID=64D21B4D69DFB3041B6375C1932BD6CB
),只要你不关闭浏览器(彻底关闭,关闭选项卡不算),浏览器会从第二次向服务器发出请求开始,一直带上这个键值对,发给服务器。服务器就会知道,这是同一个人(同一个会话)发起的请求。
(5)我们再注意一下:request.setAttribute("sysuser",userInfo)这句话:
当你第一次请求服务器时,这句代码会根据服务器默认产生的session得到ID,并与sysuser=userInfo这个键值对挂上钩(当然,userInfo可以是任何对象),保证唯一关联,检测用户是否登录就是这样实现的。
我一定要声明一点:保持一个会话与用户是否登录是没有任何关系的。
(6)再次引深一下,如果你用的不是浏览器,比如说做J2ME开发,怎样保持会话呢?
(1) 在你写完这行代码后:HttpConnection hc = (HttpConnection)Connector.open(httpURL),加入以下代码:( Constant.sessionID只是一个静态变量 )
view plaincopy to clipboardprint?
·········10········20········30········40········50········60········70········80········90········100·······110·······120·······130·······140·······150
- //在与服务器通信前设置sessionId,维持唯一的一个会话
- if (Constant.sessionID != null) {
- hc.setRequestProperty("Cookie", AppContext.CurrentAppContext.sessionID);
- }
[java] view plaincopyprint?
- //在与服务器通信前设置sessionId,维持唯一的一个会话if (Constant.sessionID != null) { hc.setRequestProperty("Cookie", AppContext.CurrentAppContext.sessionID);}
//在与服务器通信前设置sessionId,维持唯一的一个会话if (Constant.sessionID != null) { hc.setRequestProperty("Cookie", AppContext.CurrentAppContext.sessionID);}
(2) A:只向服务器读数据,不向服务写数据,B:先向服务器写数据,再从服务器读数据
对于这两种情况,只要你第一次打开openDataInputStream(),这可以加入以下代码( Constant.isLogin 只是一个静态变量boolean ):
view plaincopy to clipboardprint?
·········10········20········30········40········50········60········70········80········90········100·······110·······120·······130·······140·······150
- //每次与服务器通信后,保存 sessionId
- String cookie = hc.getHeaderField("Set-Cookie");
- if (cookie != null) {
- String jsessionId = cookie.substring(0,cookie.indexOf(";"));
- if(Constant.sessionID != null && !Constant.sessionID.equals(jsessionId) && Constant.isLogin ){
- Log.info("sessionid超时, will get new sessionid, but you must login again");
- //设置为未登录状态
- Constant.isLogin = false;
- }
- Constant.sessionID = jsessionId;
- }
[java] view plaincopyprint?
- //每次与服务器通信后,保存 sessionIdString cookie = hc.getHeaderField("Set-Cookie");if (cookie != null) { String jsessionId = cookie.substring(0,cookie.indexOf(";")); if(Constant.sessionID != null && !Constant.sessionID.equals(jsessionId) && Constant.isLogin ){ Log.info("sessionid超时, will get new sessionid, but you must login again"); //设置为未登录状态 Constant.isLogin = false; } Constant.sessionID = jsessionId;}
//每次与服务器通信后,保存 sessionIdString cookie = hc.getHeaderField("Set-Cookie");if (cookie != null) { String jsessionId = cookie.substring(0,cookie.indexOf(";")); if(Constant.sessionID != null && !Constant.sessionID.equals(jsessionId) && Constant.isLogin ){ Log.info("sessionid超时, will get new sessionid, but you must login again"); //设置为未登录状态 Constant.isLogin = false; } Constant.sessionID = jsessionId;}
这样就可以保持一个会话了。
(7)最后,关于URL重定向
引用一段话:sun帮我们想到了,所以提供了2个方法来使事情变得简单:response.encodeURL()和response.encodeRedirectURL()。这2个方法会判断cookie是否可用,如果禁用了会解析出url中的jsessionid,并连接到指定的url后面,如果没有找到jessionid会自动帮我们生成一个。至于为什么要有2个方法?这2个方法有什么不同?google了一下,说是这2个方法在判断是否要包含jsessionid的逻辑上会稍有不同。在调用 HttpServletResponse.sendRedirect前,应该先调用encodeRedirectURL()方法,否则可能会丢失 Sesssion信息。这2个方法的使用方法如:response.sendRedirect(response.encodeURL("/myapp /input.jsp"));。如果cookie没有禁用,我们在浏览器地址栏中看到的地址是这样的:/myapp/input.jsp,如果禁用了 cookie,我们会看到:/myapp /input.jsp;jsessionid=73E6B2470C91A433A6698C7681FD44F4。所以,我们在写web应用的时候,为了保险起见,应该在程序里的每一个跳转url上都使用这2个方法,来保证session的可用性。
jsessionid相关推荐
- 为什么会有jsessionid,这个东东有什么用呢?
2019独角兽企业重金招聘Python工程师标准>>> 为什么会有jsessionid,这个东东有什么用呢? 博客分类: java 为什么会有jsessionid,这个东东有什么用呢 ...
- tomcat配备禁止url显示jsessionid
2019独角兽企业重金招聘Python工程师标准>>> 我的tomcat版本是apache-tomcat-7.0.42.tar.gz 修改tomcat的conf/web.xml,在& ...
- 对于url出现jsessionid问题
当时用<c:redirect>进行页面重定向时,浏览器地址栏会出现jsessionid,这是因为redirect会去检查是否有cookie,如果没有则url上添加jsessionid以便浏 ...
- Apache Shiro去掉URL中的JSESSIONID
最近集成框架用到shiro碰到url有时候会带上jsessionid有时候又没有.以前也碰到但是没有深入研究. 网上查了半天各种方法用了都没用.比如web.xml里面加session-config,添 ...
- Spring MVC中jsessionid所引起的问题 和解决
转自:http://blog.csdn.net/seakingwy/article/details/1933687 jsessionid所引起的问题 在Spring MVC当使用Redirect ...
- Tomcat 修改JSESSIONID
2019独角兽企业重金招聘Python工程师标准>>> Tomcat在没有做任何特殊配置的情况下(默认下载包),其session的CookieID为 JSESSIONID(sessi ...
- JSESSIONID的简单说明
1)第一次访问服务器的时候,会在响应头里面看到Set-Cookie信息(只有在首次访问服务器的时候才会在响应头中出现该信息) . 上面的图JSESSIONID=ghco9xdnaco31gmafukx ...
- springboot整合shiro地址栏JSESSIONID问题
第二次情形好了,就没有了sessionID 1.client向server发送请求http://localhost:8081 2.server端经由shiro进行内部内部URL重定向至http://l ...
- Chrome开发者工具Network标签页中观察到的set-cookie jsessionid是什么东西
在Chrome开发者工具里经常能观察到HTTP响应里包含的字段: Set-Cookie: JSESSIONID= XXX 之后,就能在客户端cookie里观察到JSESSIONID=后面的值. 根据G ...
最新文章
- jquery 使用animate来改变高度自动添加样式overflow:hidden的问题
- Android10加入APEX目的
- Github Page搜索工具更新 - 收藏,手气不错
- 使用百度云OCR识别文字
- 汇编编程:在屏幕中间分别显示绿色,绿底红色,白色蓝底的字符串’welcome to masm!’
- 20190421 工作周记录-反思
- CAD2006注册机不能显示激活码
- 职场Word使用技巧大全,太实用了
- python、pygame开发的太空大战游戏源代码
- KDF- key derivation function
- 单目标跟踪CVPR 2018 ECO+
- 【内网安全】——数据库提权姿势
- 【iOS】—— KVC与KVO
- [辩论]以成败轮英雄是可取的——正方一辩稿
- @sequencegenerator oracle,SequenceGenerator注解的使用
- 我访问了一个「假」的 GitHub,难道是 PronHub ,还是 GayHub ?
- 通过DIVA了解APP安全问题
- 小学语文1-6年级心田花开成语用法归类汇总
- mysql函数之日期函数
- 对谈 | “计算机学神”闫令琪:我不希望别人说Rendering已经没东西可解了
热门文章
- JUC锁-框架与基本概念(一)
- 算法训练营08-分治和回溯
- HTML中引入CSS的三种方式——响应式Web系列学习笔记
- 中国剩余定理(孙子定理)(精华详细版!)
- Web前端开发笔记——第三章 CSS语言 第六节 CSS定位
- php 获取路由参数,路由参数 · ThinkPHP5.0完全开发手册 · 看云
- 实验详解——parted单磁盘分区并进行配额
- python online course_python-选课系统
- java形状_形状等于()
- python server.py_python manage.py runserver报错