会话状态保持,JSESSIONID,COOKIE,URL重写

居然有3W的访问量,好,我就把session和cookie的关系先来个总结,注意,是最最简单直白明了的总结了。

http协议,协议,协议,重要的说3遍,http协议主要有2大块,请求头和请求体,cookie在http请求头里,就是一个由多个KV组成的的字符串而已,其中有一个key=JSESSIONID。

client---the 1st time--->sever,header 的cookie中无JSSIONID

server--response--->client, server发现请求头中无JSSIONID,好,我来给你产生一个随机值,返回给你,在response的header中,多了一个东西, Set-Cookie:JSESSIONID=xxx

client--reveive response from server,发现response header中有Set-Cookie,好,以后我再请求server时,把这些Cookie都带上,其中就包含了JSESSIONID

client--the 2nd time--->server, 带上了所有的cookie内容

server---response--->client,带上了所有的cookie内容

所以,基于session的会话保持,是靠client和server共同协作才能完成的,server产生sessionId,然后大家一直传来传去,就这么一直玩下去,直到session timeout,或注销。


今天说说:会话状态保持,JSESSIONID,COOKIE之间的关系

在服务器端,我们用惯了session.setAttribute("",userInfo)这样的一行代码,估计你很少想到:服务器与浏览器之间是如何保持会话状态的。好了,先引用一些文章的精彩片段:

 

http://www.xxx.com/xxx_app;jsessionid=xxxxxxxxxx?a=x&b=x。

这跟一般的url基本一样,只有一个地方有区别,那就是“;jessionid=xxxxxxxx”。这个参数有时候有,有时候又没有,说它是参数可又跟一般传递的参数不同,它是紧跟在url后面用分号来分隔的,用一般的request.getParameter()方法还取不到jsessionid

 

启动你的tomcat,打开FireFox(爱得不得了,一定要安装FireBug),输入localhost就行,打开firebug,点网络,你会看到,浏览器与服务器会话的信息,给出浏览器

(1)第一次请求服务器:

浏览器的请求头信息

Hostlocalhost
User-AgentMozilla/5.0 (Windows; U; Windows NT 5.1; zh-CN; rv:1.9.2.6) Gecko/20100625 Firefox/3.6.6
Accepttext/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Languagezh-cn,zh;q=0.5
Accept-Encodinggzip,deflate
Accept-CharsetGB2312,utf-8;q=0.7,*;q=0.7
Keep-Alive115
Connection

keep-alive

 


在下图



在下图



服务器响应头信息

ServerApache-Coyote/1.1
Set-CookieJSESSIONID=64D21B4D69DFB3041B6375C1932BD6CB; Path=/
Content-Typetext/html;charset=UTF-8
Content-Languagezh-CN
Content-Length242
DateMon, 28 Jun 2010 02:35:29 GMT

(2)第二次请求服务器:

浏览器的请求头信息

Hostlocalhost
User-AgentMozilla/5.0 (Windows; U; Windows NT 5.1; zh-CN; rv:1.9.2.6) Gecko/20100625 Firefox/3.6.6
Accepttext/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Languagezh-cn,zh;q=0.5
Accept-Encodinggzip,deflate
Accept-CharsetGB2312,utf-8;q=0.7,*;q=0.7
Keep-Alive115
Connectionkeep-alive
CookieJSESSIONID=64D21B4D69DFB3041B6375C1932BD6CB

服务器响应头信息

ServerApache-Coyote/1.1
Content-Typetext/html;charset=UTF-8
Content-Languagezh-CN
Content-Length242
DateMon, 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只是一个静态变量)

 

(2)A:只向服务器读数据,不向服务写数据,B:先向服务器写数据,再从服务器读数据

对于这两种情况,只要你第一次打开openDataInputStream(),这可以加入以下代码(Constant.isLogin只是一个静态变量boolean):

 

这样就可以保持一个会话了。


(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的可用性。

原文引自

 

(8)Fighting And Keep Moving!!

  • 11
    点赞
  • 40
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Thymeleaf允许在应用程序中配置URL重写过滤器,它通过调用Thymeleaf模板生成的每个URL的Servlet API的javax.servlet.http.HttpServletResponse类中的response.encodeURL()方法来实现。要使用URL重写,需要在Web应用程序中配置一个过滤器,该过滤器将在生成的HTML中重写URL。以下是一个简单的示例,演示如何在Spring Boot应用程序中配置Thymeleaf URL重写过滤器: 1.在pom.xml文件中添加以下依赖项: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-thymeleaf</artifactId> </dependency> ``` 2.在application.properties文件中添加以下配置: ```properties server.servlet.context-path=/myapp server.tomcat.url-encoding=UTF-8 ``` 3.创建一个名为UrlRewriteFilter的Java类,该类实现了javax.servlet.Filter接口,并在doFilter()方法中调用response.encodeURL()方法: ```java import javax.servlet.*; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; public class UrlRewriteFilter implements Filter { @Override public void init(FilterConfig filterConfig) throws ServletException { } @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { HttpServletRequest req = (HttpServletRequest) request; HttpServletResponse resp = (HttpServletResponse) response; String url = req.getRequestURI().substring(req.getContextPath().length()); String rewrittenUrl = resp.encodeURL(url); chain.doFilter(request, new UrlRewriteResponseWrapper(resp, rewrittenUrl)); } @Override public void destroy() { } } ``` 4.创建一个名为UrlRewriteResponseWrapper的Java类,该类扩展了javax.servlet.http.HttpServletResponseWrapper类,并重写了encodeURL()方法: ```java import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpServletResponseWrapper; import java.io.IOException; public class UrlRewriteResponseWrapper extends HttpServletResponseWrapper { private String rewrittenUrl; public UrlRewriteResponseWrapper(HttpServletResponse response, String rewrittenUrl) { super(response); this.rewrittenUrl = rewrittenUrl; } @Override public String encodeURL(String url) { return rewrittenUrl; } @Override public String encodeRedirectURL(String url) { return rewrittenUrl; } @Override public String encodeUrl(String url) { return rewrittenUrl; } @Override public String encodeRedirectUrl(String url) { return rewrittenUrl; } @Override public void sendRedirect(String location) throws IOException { super.sendRedirect(rewrittenUrl); } } ``` 5.在Spring Boot应用程序的配置类中注册UrlRewriteFilter过滤器: ```java import org.springframework.boot.web.servlet.FilterRegistrationBean; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @Configuration public class AppConfig { @Bean public FilterRegistrationBean<UrlRewriteFilter> urlRewriteFilter() { FilterRegistrationBean<UrlRewriteFilter> registrationBean = new FilterRegistrationBean<>(); registrationBean.setFilter(new UrlRewriteFilter()); registrationBean.addUrlPatterns("/*"); return registrationBean; } } ``` 这样,Thymeleaf URL重写过滤器就配置完成了。在Thymeleaf模板中,可以使用th:href属性来生成URL,如下所示: ```html <a th:href="@{/hello}">Hello</a> ``` 这将生成一个相对于应用程序上下文路径的URL,例如/myapp/hello。当用户单击链接时,UrlRewriteFilter过滤器将调用response.encodeURL()方法来重写URL,以便在会话ID中包含JSESSIONID参数(如果需要)。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值