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{}
第三種加鎖非常影響性能,因此一般使用第二種方法。