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{}
第三种加锁非常影响性能,因此一般使用第二种方法。
网友评论