Servlet
Servlet默认是单实例多线程的。在第一个请求到来时进行初始化。对每一个请求从线程池中取一个线程进行响应。Servlet不是线程安全的。
public interface Servlet {
void init(ServletConfig var1) throws ServletException;
ServletConfig getServletConfig();
void service(ServletRequest var1, ServletResponse var2) throws ServletException, IOException;
String getServletInfo();
void destroy();
}
- SpringMVC中的DispatcherServlet
<servlet>
<servlet-name>chapter2</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>chapter2</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
Filter
一个简单的filter如下,该filter的任务是把description从tomcat默认的iso8859-1转成UTF-8,以正确处理字符串。
显然,我们的任务是拦截ServletRequest servletRequest,修改之,然后让该servletRequest继续向下处理。
public class TransCharSetFilter implements Filter {
private Logger logger = LoggerFactory.getLogger(TransCharSetFilter.class);
@Override
public void init(FilterConfig filterConfig) throws ServletException {
logger.info("TransCharSetFilter init");
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
logger.info("into filter");
if(servletRequest.getParameter("description")!=null){
String afterTrans = new String(servletRequest.getParameter("description").getBytes("iso8859-1"),"UTF-8");
//写到这里,我们发现ServletRequest并没有setParameter方法,被逼无奈,我们只能自己实现一个装饰器模式
HttpServletRequestWrapperModify modify = new HttpServletRequestWrapperModify((HttpServletRequest)servletRequest);
filterChain.doFilter(modify,servletResponse);
}
//转发到后续filter
filterChain.doFilter(servletRequest,servletResponse);
//在这之后,我们可以对servletResponse进行修改。大概方法和修改servletRequest相同。
//修改完成后,使用如下代码替换输出
/*
ServletOutputStream out = response.getOutputStream();
out.write(XXX);
out.flush();
*/
}
@Override
public void destroy() {
logger.info("TransCharSetFilter destroy");
}
}
相关类图如下所示
我们将实现一个类
public class HttpServletRequestWrapperModify extends HttpServletRequestWrapper {
private Map<String, String> parameterMap = new HashMap<>();
public HttpServletRequestWrapperModify(HttpServletRequest request) {
super(request);
}
@Override
public Enumeration<String> getParameterNames() {
Vector<String> vector = new Vector<>(this.getParameterMap().keySet());
for (String key : parameterMap.keySet()){
if(!vector.contains(key)){
vector.add(key);
}
}
return vector.elements();
}
@Override
public String getParameter(String name) {
if(parameterMap.containsKey(name)){
return parameterMap.get(name);
}else{
return this.getParameter(name);
}
}
}
最后完成配置,大功搞成!
<filter-mapping>
<filter-name>TransCharSetFilter</filter-name>
<url-pattern>/remoting/*</url-pattern>
</filter-mapping>
<welcome-file-list>
<welcome-file>index.jsp</welcome-file>
</welcome-file-list>
Listener
SpringMVC中的Interceptor
基于Spring AOP
和filter有如下区别:
- filter的过滤范围比interceptor大,filter除了过滤请求外通过通配符可以保护页面、图片、文件等,而interceptor只能过滤请求,只对action起作用,在action之前开始,在action完成后结束(如被拦截,不执行action);