Tomcat 7源码学习笔记 -6 encodeURL深度解析补充

之前提到调用response.encodeURL(String url)方法的时候,如果客户端支持cookie,那么不做任何处理,直接返回传进去的url,因为jsessionid可以写入cookie中,所以不需要url重写。其实所说的客户端支持cookie并不严谨,因为在客户端支持cookie的情况下,调用response.encodeURL(String url)方法,也会遇到对传进去的url进行重写的情况。那么什么情况下会出现这个问题呢?请看下面:

客户端支持cookie,第一次访问某个servlet,这时servlet接收到的request的cookie中没有jsessionid信息,这时encodeURL内部会认为客户端不支持cookie,所以会重写传进去的url,即使你在请求url中手动添加了正确的;jsessionid=30A7F107A1E5B1E7250622C158244032信息,也仍然会重写传进去的url。

由此可以得出encodeURL内部的具体的判定规则:

request的cookie中是否有正确的jsessionid信息。

有的话,就不重写传进去的url。否则符合重写url的条件(当然还有其他条件也必须符合)。

从代码中也可以得出相同的结论:

boolean isEncodeable(final String location)方法中

if (hreq.isRequestedSessionIdFromCookie()) {
        return (false);
}

isRequestedSessionIdFromCookie()方法如下:

public boolean isRequestedSessionIdFromCookie() {

        if (requestedSessionId == null) {
            return false;
        }

        return requestedSessionCookie;
}

 requestedSessionId是指从客户端得到的jsessionid的值,具体采用cookie传递还是url传递都没关系。

1.如果requestedSessionId为空,说明客户端没有传入jsessionid,当然cookie中也没有,所以符合url重写的条件。

2.如果requestedSessionId有值,再进一步判断这个jsessionid是否是从cookie中取到的,如果是那么就不用重写url,否则重写。

上面这些分析可以看出:

1.servlet或者说tomcat其实是无法准确地获知浏览器是否禁用了cookie。

2.即使浏览器禁用了cookie,只要servlet里面创建了session,那么jsessionid同样会写入response的cookie,并返回给客户端,只是浏览器不往本地cookie中写,直接忽略掉了。

3.只要在request的cookie中没有找到jsessionid,那么就可以进行url重写。

最后,顺便看一下HttpServletRequest中的下面这几个方法:

public String getRequestedSessionId() //获取客户端传过来的jsessionid的值

public boolean isRequestedSessionIdFromCookie() //request的cookie中是否有jsessionid

public boolean isRequestedSessionIdFromURL() //request url中是否有jsessionid

那么,有一个问题,如果客户端的cookie中含有jsessionid,同时request url中也含有jsessionid,这时候会怎么样呢?

答案是:使用cookie中的jsessionid

1.优先判断cookie中有没有,cookie中有的话,就使用cookie中的jsessionid

2.如果cookie中没有,再看request url中有没有

isRequestedSessionIdFromCookie()返回 true

isRequestedSessionIdFromURL()返回 false

相关推荐