Servlet单实例多线程

1

Servlet容器为了提高响应速度,默认只允许同一个serlvet一个实例存在,对于多个请求使用多线程的方式来处理,避免了实例化servlet时的开销,当有请求到来时,servlet容器会从线程池中取一个空闲的线程,来处理请求,处理完毕后即放回线程池。
这样对于serlvet的实例变量在使用时需要注意下,实例变量并不是线程安全的,局部变量才是。下面看一个例子。

public class TestServlet extends HttpServlet {
    private static final long serialVersionUID = 1L;
       private String message;
    /**
     * @see HttpServlet#HttpServlet()
     */
    public TestServlet() {
        super();
        // TODO Auto-generated constructor stub
    }

    /**
     * @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response)
     */
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // TODO Auto-generated method stub
        message = request.getParameter("message");
        String oldMessage = message;
        Thread cur = Thread.currentThread();
        System.err.println("当前threadId="+cur.getId()+" message="+message);
        try {
            Thread.sleep(5000);
        } catch (InterruptedException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        System.err.println("oldMesssage="+oldMessage+" message = "+message);
    }

    /**
     * @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response)
     */
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        // TODO Auto-generated method stub
    }

}

首先访问

http://127.0.0.1:8192/ServletTest/TestServlet?message=123

然后访问

http://127.0.0.1:8192/ServletTest/TestServlet?message=456

可以看到控制台输出:

当前threadId=20 message=123
当前threadId=21 message=456
oldMesssage=123 message = 456
oldMesssage=456 message = 456

分析结果可以得知,第二次访问时修改了实例变量message,导致最终第一次访问oldMessage和message不一致。
解决方法有三个:
1 使用Javax.servlet.SingleThreadModel
不过这个已经废弃了,不再多说。
2 去掉实例变量 使用局部变量。
3 使用同步 synchronized{}
第三种加锁非常影响性能,因此一般使用第二种方法。

推荐阅读更多精彩内容