不少初學者,甚至是工做1-3年的小夥伴們均可能弄不明白?servlet Struts1 Struts2 springmvc 哪些是單例,哪些是多例,哪些是線程安全?java
在談這個話題以前,咱們先了解一下Java中相關的變量類型以及內存模型JMM。web
Java的內存模型JMM(Java Memory Model)JMM主要是爲了規定了線程和內存之間的一些關係。根據JMM的設計,系統存在一個主內存(Main Memory),Java中全部實例變量都儲存在主存中,對於全部線程都是共享的。每條線程都有本身的工做內存(Working Memory),工做內存由緩存和堆棧兩部分組成,緩存中保存的是主存中變量的拷貝,緩存可能並不總和主存同步,也就是緩存中變量的修改可能沒有馬上寫到主存中;堆棧中保存的是線程的局部變量,線程之間沒法相互直接訪問堆棧中的變量。根據JMM,咱們能夠將論文中所討論的Servlet實例的內存模型抽象爲下圖所示的模型。spring
Servlet/JSP技術和ASP、PHP等相比,因爲其多線程運行而具備很高的執行效率。因爲Servlet/JSP默認是以多線程模式執行的,因此,在編寫代碼時須要很是細緻地考慮多線程的安全性問題。然而,不少人編寫Servlet/JSP程序時並無注意到多線程安全性的問題,這每每形成編寫的程序在少許用戶訪問時沒有任何問題,而在併發用戶上升到必定值時,就會常常出現一些莫明其妙的問題。緩存
Servlet體系結構是創建在Java多線程機制之上的,它的生命週期是由Web容器負責的。當客戶端第一次請求某個Servlet 時,Servlet容器將會根據web.xml配置文件實例化這個Servlet類。當有新的客戶端請求該Servlet時,通常不會再實例化該 Servlet類,也就是有多個線程在使用這個實例。Servlet容器會自動使用線程池等技術來支持系統的運行,以下圖所示。安全
這樣,當兩個或多個線程同時訪問同一個Servlet時,可能會發生多個線程同時訪問同一資源的狀況,數據可能會變得不一致。因此在用Servlet構建的Web應用時若是不注意線程安全的問題,會使所寫的Servlet程序有難以發現的錯誤。多線程
Servlet的線程安全問題主要是因爲實例變量使用不當而引發的,這裏以一個現實的例子來講明。併發
/** * 模擬用戶AB在同時執行不一樣的動做 * 先執行 http://localhost:8080/concurrent?username=A&action=play * 稍後執行 http://localhost:8080/concurrent?username=B&action=eat */ public class Concurrent extends HttpServlet { private static final long serialVersionUID = 1L; private String action = "";//動做 public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { try { String username = request.getParameter("username"); action = request.getParameter("username"); Thread.sleep(5000); //爲了突出併發問題,在這設置一個延時 //若是不出意外,應該用戶AB都在吃飯 System.out.println("用戶:"+username+"在"+action); } catch (Exception e) { } } }
首先,明確一點Sturts1 action是單例模式,線程是不安全的。Struts1使用的ActionServlet是單例的,既然是單例,當使用實例變量的時候就會有線程安全的問題。全部通常在開發中試禁止使用實例變量的。mvc
struts2使用的是actionContext,都是使用裏面的實例變量,讓struts2自動匹配成對象的。每次處理一個請求,struts2就會實例化一個對象,這樣就不會有線程安全的問題了。性能
須要注意的是,若是struts2+spring來管理注入的時候,不要把Action設置成單例,不然會出問題的。固然如今不多有項目使用struts2了。線程
SpringMVC的controller默認是單例模式的,因此也會有多線程併發的問題。
servlet Struts1 SpringMvc 是線程不安全的,固然若是你不使用實例變量也就不存在線程安全的問題了。
Struts2 是線程安全的,固然前提狀況是,Action 不交給 spring管理,而且不設置爲單例。
SpringMvc 的 Bean 能夠設置成多例變成線程安全,可是必定程度上回影響系統性能。