Filter的线程安全问题

Filter的线程安全问题:
马克- to-win:马克 java社区:防盗版实名手机尾号: 73203。
马克-to-win:和Servlet一样,为了提高性能,Filter也采取多线程模式。即:每一个线程来应答一个用户浏览器,而且这个线程和用户要访问的目标Servlet的线程是同一个线程。说得更准确一点,当用户访问某个资源需要经过过滤器时,服务器中一个线程为了应答这个客户请求,先调用过滤器中的doFilter方法,再根据是否有chain.doFilter的指令,决定是否调用目标资源的doXXX方法,当然肯定还由这同一个线程调用。马克 -to-win:执行完doXXX方法以后,自然要继续完成doFilter方法里面chain.doFilter语句后面的语句,这就和基本的C语言里主程序调用子程序的规则一样了。



例 1.2.6

package com;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class ServletHello1 extends HttpServlet {
    protected void doGet(HttpServletRequest request, HttpServletResponse response) {
         System.out.println(Thread.currentThread().getName()+"hello");
    }
}




package com;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.*;
public class MarkToWinFilter implements Filter {
    public MarkToWinFilter() {
    }
    public void destroy() {
    }
    public void doFilter(ServletRequest request, ServletResponse response,
            FilterChain chain) throws IOException, ServletException {
        response.setCharacterEncoding("GBK");
        System.out.println(Thread.currentThread().getName()+"之前 filterChain.doFilter");
        chain.doFilter(request, response);
        System.out.println(Thread.currentThread().getName()+"之后 filterChain.doFilter");
    }
    public void init(FilterConfig fConfig) throws ServletException {
    }
}




实验:

设备:这个实验我是用三个浏览器一起做的:一个是火狐浏览器,一个是360浏览器,另外一个是eclipse自带的内置浏览器。

方法:访问的顺序是,火狐一次,360一次,内置浏览器一次,火狐最后一次。


实验输出结果:



http-8080-Processor25之前 filterChain.doFilter
http-8080-Processor25hello
http-8080-Processor25之后 filterChain.doFilter

http-8080-Processor22之前 filterChain.doFilter
http-8080-Processor22hello
http-8080-Processor22之后 filterChain.doFilter

http-8080-Processor23之前 filterChain.doFilter
http-8080-Processor23hello
http-8080-Processor23之后 filterChain.doFilter

http-8080-Processor24之前 filterChain.doFilter
http-8080-Processor24hello
http-8080-Processor24之后 filterChain.doFilter



实验结论,当用一个浏览器访问时,系统会用同一个线程来运行filter和servlet。但线程和浏览器没有对应关系。所以Filter的类变量和 servlet类变量一样,都有线程安全问题。有关线程安全问题和解决方案请参考Servlet的线程安全。(我的前面jsp第二章)