看RequestContextListener和RequestContextHolder源代码不难看出,是用过ThreadLocal实现的。
ThreadLocal是什么呢?看词义,它并非是一个线程,它不是一个Thread,而是一个线程局部变量,其实功能非常简单,就是为每一个使用该变量的线程都提供一个变量值的副本,是每一个线程都可以独立地改变自己的副本,而不会和其它线程的副本冲突。从线程的角度看,就好像每一个线程都完全拥有该变量。
每一个HTTP请求、都是一个独立的线程,有一个独立的ThreadLocal。利用该特性我们可以利用ThreadLocal进行、HTTP请求生命周期的暂时存取值,不同类之间进行传值。
而RequestContextListener就是通过这一特性实现的。
操作方法如下:
第一步:在web.xml中,加入如下监听
<!-- 监听器 -->
<listener>
<listener-class>org.springframework.web.context.request.RequestContextListener</listener-class>
</listener>
第二步:写一个静态方法获取request
package util;
import javax.servlet.http.HttpServletRequest;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
//在Java任何位置获取request对象 需要在web.xml中配置监听器
public class RequestUtil {
public static HttpServletRequest getRequest(){
try{
return ((ServletRequestAttributes)RequestContextHolder.getRequestAttributes()).getRequest();
} catch(Exception e){
return null;
}
}
}
这样在web请求过程中,在任何地方都可以调用这个静态方法获取request对象了。
在spring中请求如下req.do
package controller;
import javax.servlet.http.HttpServletRequest;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import util.RequestUtil;
@Controller
public class ReqController {
@RequestMapping("/req.do")
public String test(){
HttpServletRequest re = RequestUtil.getRequest();//获取request
System.out.println(re.getClass());
String path = re.getContextPath();
System.out.println(path);
return "req/result";
}
}
在Controller中获取request