JShop简介:jshop是一套使用Java语言开发的B2C网店系统,致力于为个人和中小企业提供免费、好用的网店系统。

项目主页:http://git.oschina.net/dinguangx/jshop

在线演示:

在spring mvc中,为了随时都能取到当前请求的request对象,可以通过RequestContextHolder的静态方法getRequestAttributes()获取Request相关的变量,如request, response等。 
        在jshop中,对RequestContextHolder的使用进一步封装,简化为RequestHolder类,如下:

public class RequestHolder {

public static HttpServletRequest getRequest(){

HttpServletRequest req = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();

return req;

}

<span class="hljs-keyword">public <span class="hljs-keyword">static HttpServletResponse <span class="hljs-title">getResponse(){</br>
HttpServletResponse resp = ((ServletWebRequest)RequestContextHolder.getRequestAttributes()).getResponse();</br>
<span class="hljs-keyword">return resp;</br>
}</br></br>

}

在大部分的情况下,它都能很好地工作,但在商品管理编辑中,新增商品时,却出现了意外的问题:通过RequestHolder.getRequest().getParameter()得不到参数值,通过debug发现,
通过spring mvc的method注入的request对象实际为MultipartHttpServletRequest,而通过RequestHolder.getRequest()获取到的request对象却是org.apache.catalina.connector.RequestFacade的实例。

public class RequestFacade implements 
HttpServletRequest

原来在商品新增时,由于使用了文件上传,form表单的enctype类型为”multipart/form-data”,
spring mvc对文件上传的处理类实际却为spring-mvc.xml文件中配置的CommonsMultipartResolver
该类先判断当前请求是否为multipart类型,如果是的话,将request对象转为MultipartHttpServletRequet,相关的源码见DisptcherServlet


protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {

HttpServletRequest processedRequest = request;

......
processedRequest = checkMultipart(request);

multipartRequestParsed = processedRequest != request;


......
// Actually invoke the handler.

mv = ha.handle(processedRequest, response, mappedHandler.getHandler());

......
}
<span class="hljs-keyword">protected HttpServletRequest <span class="hljs-title"><span style="color: #ff0000; font-size: 14pt;"><strong>checkMultipart</strong></span>(HttpServletRequest request) <span class="hljs-keyword">throws MultipartException {</br>
<span class="hljs-keyword">if (<span class="hljs-keyword">this.multipartResolver != <span class="hljs-keyword">null &amp;&amp; <span class="hljs-keyword">this.multipartResolver.isMultipart(request)) {</br>
<span class="hljs-keyword">if (request <span class="hljs-keyword"><span style="color: #ff0000; font-size: 14pt;"><strong>instanceof MultipartHttpServletRequest</strong></span>) {</br>
logger.debug(<span class="hljs-string">"Request is already a MultipartHttpServletRequest - if <span style="color: #0000ff; font-size: 18pt;"><strong>not in a forward</strong></span>, " +</br>
<span class="hljs-string">"this typically results from an additional MultipartFilter in web.xml");</br>
}</br>
<span class="hljs-keyword">else {</br>
<span class="hljs-keyword">return <span class="hljs-keyword">this.multipartResolver.<span style="color: #0000ff; font-size: 14pt;"><strong>resolveMultipart</strong></span>(request);</br>
}</br>
}</br>
<span class="hljs-comment">// If not returned before: return original request.</br>
<span class="hljs-keyword">return request;</br>
}</span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></span></code></pre>

那么,RequestContextHolder中的request又是从哪来的呢? 
继续翻看DispatcherServlet的源码,从其父类FrameworkServlet中找到的processRequest()以相关方法源码:


protected final void processRequest(HttpServletRequest request, HttpServletResponse response)

throws ServletException, IOException {


......
    RequestAttributes previousAttributes = RequestContextHolder.getRequestAttributes();</br>
ServletRequestAttributes requestAttributes = buildRequestAttributes(request, response, previousAttributes);</br></br> WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);</br>
asyncManager.registerCallableInterceptor(FrameworkServlet.class.getName(), <span class="hljs-keyword">new RequestBindingInterceptor());</br></br> initContextHolders(request, localeContext, requestAttributes);</br></br> <span class="hljs-keyword">try {</br>
doService(request, response);</br>
}</br>
......</br>
}</br> <span class="hljs-keyword">protected ServletRequestAttributes <span class="hljs-title"><span style="color: #ff0000; font-size: 14pt;"><strong>buildRequestAttributes</strong></span>(</br>
HttpServletRequest request, HttpServletResponse response, RequestAttributes previousAttributes) {</br></br> <span class="hljs-keyword">if (previousAttributes == <span class="hljs-keyword">null || previousAttributes <span class="hljs-keyword"><span style="color: #ff0000; font-size: 14pt;"><strong>instanceof ServletRequestAttributes</strong></span>) {</br></br>
<span class="hljs-keyword">return <span class="hljs-keyword">new ServletRequestAttributes(request);
}</br>
<span class="hljs-keyword">else {</br>
<span class="hljs-keyword">return <span class="hljs-keyword">null; <span class="hljs-comment">// preserve the pre-bound RequestAttributes instance</br>
}</br>
}</br></br> <span class="hljs-keyword">private <span class="hljs-keyword">void <span class="hljs-title">initContextHolders(</br>
HttpServletRequest request, LocaleContext localeContext, RequestAttributes requestAttributes) {</br></br> <span class="hljs-keyword">if (localeContext != <span class="hljs-keyword">null) {</br>
LocaleContextHolder.setLocaleContext(localeContext, <span class="hljs-keyword">this.threadContextInheritable);</br>
}</br>
<span class="hljs-keyword">if (requestAttributes != <span class="hljs-keyword">null) {</br>
RequestContextHolder.setRequestAttributes(requestAttributes, <span class="hljs-keyword">this.threadContextInheritable);</br>
}</br>
<span class="hljs-keyword">if (logger.isTraceEnabled()) {</br>
logger.trace(<span class="hljs-string">"Bound request context to thread: " + request);</br>
}</br>
}</br>

从这里可以看到,initContextHolder()方法中完成了RequestContextHolder的requestAttributes设置,
而doService()在这之后调用,DispatcherServlet中的processRequest()方法即在doService()之中,
所以从RequestContextHolder中获取到的就是原来的RequestFacade对象,而不是经过spring mvc处理之后的MultipartHttpServletRequest对象,
其后果就是,从RequestContextHolder获取request后,无法直接通过getParameter()获取参数值。

最便捷的解决办法: 
直接将HttpServletRequest作为spring mvc的方法入参,即可以正确获取参数值

Jshop简介:http://git.oschina.net/dinguangx/jshop

http://dinguangx.iteye.com/blog/2227049

将登录等信息保存到session中和退出session的更多相关文章

  1. 将用户信息保存到Cookie中

    /** * 把用户保存到Cookie * * @param request * @param response * @param member */ private void rememberPwdA ...

  2. SWIFT中将信息保存到plist文件内

    在项目中可能我们需要保存一些数据到plist文件中,以下就本人在学习过程中的笔记,不成熟的地方请指出. 可能我有一个类叫做Student import UIKit class Student: NSO ...

  3. 爬取豆瓣电影信息保存到Excel

    from bs4 import BeautifulSoup import requests import html.parser from openpyxl import Workbook,load_ ...

  4. DOS命令将黑框中查询到的信息保存到TXT等文件里

  5. python爬取当当网的书籍信息并保存到csv文件

    python爬取当当网的书籍信息并保存到csv文件 依赖的库: requests #用来获取页面内容 BeautifulSoup #opython3不能安装BeautifulSoup,但可以安装Bea ...

  6. Python scrapy爬虫数据保存到MySQL数据库

    除将爬取到的信息写入文件中之外,程序也可通过修改 Pipeline 文件将数据保存到数据库中.为了使用数据库来保存爬取到的信息,在 MySQL 的 python 数据库中执行如下 SQL 语句来创建 ...

  7. 将Prometheus alerts保存到elasticsearch

    Prometheus产生的告警通常会发送到alertmanager,当使用alertmanager时,其告警信息仅存在于alertmanager的内存中,无法持久化.故实现了小工具,用于将Promet ...

  8. Introduction and use of Cookie and Session(Cookie&Session的介绍和使用)

    一.Cookie 1.什么是Cookie? Cookie是HTTP协议的规范之一,它是服务器和客户端之间传输的小数据. 首先由服务器通过响应头把Cookie传输给客户端,客户端会将Cookie保存起来 ...

  9. 利用session_set_save_handler()函数将session保存到MySQL数据库中

    PHP保存session默认的是采用的文件的方式来保存的,这仅仅在文件的空间开销很小的windows上是可以采用的,但是如果我们采用uinx或者是liux上的文件系统的时候,这样的文件系统的文件空间开 ...

随机推荐

  1. Elasticsearch (1) - 索引库 文档 分词

    创建索引库 ES的索引库是一个逻辑概念,它包括了分词列表及文档列表,同一个索引库中存储了相同类型的文档.它就相当于MySQL中的表,或相当于Mongodb中的集合. 关于索引这个语: 索引(名词):E ...

  2. 【HEVC帧间预测论文】P1.1 基于运动特征的HEVC快速帧间预测算法

    基于运动特征的 HEVC 快速帧间预测算法/Fast Inter-Frame Prediction Algorithm for HEVC Based on Motion Features <HE ...

  3. MongoDB最简单的入门教程之二 使用nodejs访问MongoDB

    在前一篇教程 MongoDB最简单的入门教程之一 环境搭建 里,我们已经完成了MongoDB的环境搭建. 在localhost:27017的服务器上,在数据库admin下面创建了一个名为person的 ...

  4. 位(bit)、字节(byte)、字

    1.位(bit)来自英文bit,音译为“比特”,表示二进制位.位是计算机内部数据储存的最小单位,11010100是一个8位二进制数.一个二进制位只可以表示0和1两种状态(21):两个二进制位可以表示0 ...

  5. ES6 export default 和 export 的区别

    export default 和 export 区别: 1.export与export default均可用于导出常量.函数.文件.模块等 2.你可以在其它文件或模块中通过import+(常量 | 函 ...

  6. 【php】查看扩展的版本

    php --ri [扩展名称] 例如: php --ri mongodb mongodb MongoDB support => enabled MongoDB extension version ...

  7. delphi 动态调用API

    好处没有这个API 也可以启动程序只是调用会出错. function __IsNativeVhdBoot: Boolean; type TIsNativeVhdBoot = function( Nat ...

  8. django 127.0.0.1 将您重定向的次数过多

    "GET /?next=/%3Fnext%3D/%253Fnext%253D/ HTTP/1.1" 302 0 solution reference from django.con ...

  9. 不同子系统采用不同MySQL编码LATIN1和UTF8的兼容

    程序处理 这是一个历史遗留系统, 旧的系统是C++开发的, 插入数据的时候, 没有统一MYSQL各个层次(服务器, 数据库, 表, 列)的编码, 这个情况基本上是MYSQL的默认安装导致的, 实际的数 ...

  10. ubuntu 宝塔安装一条龙服务

    ubuntu 安装 1, wget -O install.sh http://download.bt.cn/install/install-ubuntu.sh && sudo bash ...