Java面試題:Servlet是執行緒安全的嗎?
阿新 • • 發佈:2018-11-09
轉自: http://www.cnblogs.com/chanshuyi/p/5052426.html
Servlet不是執行緒安全的。
要解釋為什麼Servlet為什麼不是執行緒安全的,需要了解Servlet容器(即Tomcat)使如何響應HTTP請求的。
當Tomcat接收到Client的HTTP請求時,Tomcat從執行緒池中取出一個執行緒,之後找到該請求對應的Servlet物件。如果該Servlet還未被請求過,那麼將進行Servlet初始化並呼叫Servlet並呼叫service()方法。否則,直接呼叫service()方法。要注意的是每一個Servlet物件再Tomcat容器中只有一個例項物件,即是單例模式。如果多個HTTP請求請求的是同一個Servlet,那麼著兩個HTTP請求對應的執行緒將併發呼叫Servlet的service()方法。
這時候,如果在Servlet中定義了例項變數或靜態變數,那麼可能會發生執行緒安全問題(因為所有的執行緒都可能使用這些變數)。
比如下面的Servlet中的name
和i
變數就會引發執行緒安全問題。
import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.IOException; import java.text.SimpleDateFormat; import java.util.Date; public class ThreadSafeServlet extends HttpServlet { public static String name = "Hello"; //靜態變數,可能發生執行緒安全問題 int i; //例項變數,可能發生執行緒安全問題 SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss"); @Override public void init() throws ServletException { super.init(); System.out.println("Servlet初始化"); } @Override protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.printf("%s:%s[%s]\n", Thread.currentThread().getName(), i, format.format(new Date())); i++; try { Thread.sleep(5000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.printf("%s:%s[%s]\n", Thread.currentThread().getName(), i, format.format(new Date())); resp.getWriter().println("<html><body><h1>" + i + "</h1></body></html>"); } }
在Tomcat中啟動這個Servlet並在瀏覽器發起多個HTTP訪問,最後會發現變數i
是多執行緒共享的。
如果需要更加深入透徹地瞭解Tomcat接收HTTP的細節,以及與Servlet互動的細節,可以深入看看Tomcat的架構和原始碼。
參考資料:
1、http://www.ibm.com/developerworks/cn/java/j-lo-tomcat1/
2、http://blog.csdn.net/cutesource/article/details/5040417