Spring MVC是如何逐步簡化Servlet的編程的java
Servlet和JSP是開發java Web應用程序的兩種基本技術,Spring MVC是Spring框架中用於Web應用程序開發的一個模塊,可以清晰的瞭解到從Servlet到Spring MVC開發技術之間逐步簡化的過程對於深入理解Spring MVC在Web應用程序開發中的做用具備很大的幫助。咱們會從項目的目錄結構來介紹這個簡化的過程:web
1、 在Servlet中spring
在應用程序目錄下是WEB-INF目錄,它包含classes子目錄,Servlet類以及其餘的Java類必須放在這個下面,在最基本的Servlet類中,須要實現Servlet接口定義的init(),service(),destroy(),getServletConfig()和getServletInfo()方法,其中最基本的邏輯放在service()方法中進行編寫,在servic()方法中最經常使用的是會經過PrintWriter進行內容的輸出。數據庫
演進1:實現Servlet接口的時候必須將全部的方法進行實現,即使有些根本沒有包含任何代碼。可是GenericServlet抽象類實現了Servlet和ServletConfig接口簡化了任務。express |
所以在GenericServlet抽象類的幫助下,咱們只須要重寫service方法中實現咱們的任務就能夠了。編程
演進2:然而GenericServlet並不經常使用,由於HttpServlet纔是主角,而且不須要覆蓋service()方法而是doGet(),doPost()來編寫邏輯。後端 |
HttpServlet覆蓋了GenericServlet類,它將ServletRequest和ServletRespond對象分別轉換成了HttpServletRequest和HttpServletRespond對象,並調用最經常使用的doGet()(從服務器端向客戶端呈現),doPost()(從客戶端得到到服務器端處理)等七種方法而不須要重寫service方法。api
利用部署描述符是一種配置Servlet應用程序的方法,部署描述符命名爲web.xml並放在WEB-INF目錄下。 瀏覽器
Servlet還提供了四種狀態保持技術:URL重寫,隱藏域,cookies和HTTPSession。其中HTTPSession是最經常使用的。tomcat
2、JSP的加入
演進3:Servlet有兩個缺點:1)寫在Servlet中的全部HTML標籤必須包含java字符串似的處理HTTP響應報文工做複雜;2)全部的文本都是硬編碼,便是出現了一點點的變化也須要重現編譯。JSP解決了上述的問題並與Servlet同時使用。 |
JSP本質上是一個Servlet,然而其不須要編譯,JSP頁面是一個以.jsp擴展名的文本文件。簡單的JSP頁面在第一次請求後被翻譯爲(JSP名)_jsp的Servlet,翻譯以後的Servelt能夠看到:_jspInit(),_jspDestory(),_jspService()這樣的方法其實都是和Servlet相對應的。
放在WEB-INF文件夾下的內容是沒法直接經過瀏覽器輸入地址訪問的,而WEB-INF文件夾外的則是能夠的,而且添加了新的JSP頁面後無需重啓JSP/Servlet容器(如tomcat)。
解耦1:使用標準JSP訪問,操做JavaBean,是實現展示(HTML)與業務實現(Java 代碼)分離的第一步。 |
JSP中的註釋(<%-- 內容 -- %>),指令(<%@ 指令名 %>),腳本(<% 腳本 %>),表達式(<%= 賦值結果 %>),聲明(<%! 聲明 %>),動做(<jsp:useBean />)。
演進4:JSP中的EL能夠輕鬆訪問應用程序數據,使得JSP頁面不須要任何的聲明,表達式和腳本。 |
EL表達式${expression}以及取值[]和.運算符。
演進5:JSP標準標籤庫(JSTL)在EL的基礎上進一步解決了遍歷Map,集合,條件測試,XML處理,數據庫操做訪問等操做的問題。 |
使用JSTL須要taglib指令:
<%@ taglib uri=「uri」 prefix=「prefix」 %>
JSTL標籤類型:聲明賦值,條件判斷,循環遍歷,格式化,函數(主要是字符串函數)
演進6:JSP標準標籤庫(JSTL)提供了一些標籤能解決經常使用的問題,可是對於一些很是見惡問題,須要擴展javax.servlet.jsp.tagetx包中的成員實現自定義標籤。 |
自定義標籤的實現,叫做標籤處理器,而簡單標籤處理器是指繼承SimpleTag實現的經典自定義標籤。經典標籤處理器須要實現Tag,IterationTag及BodyTag接口或者擴展TagSupport,BodyTagSupport兩個類;簡單標籤處理器須要實現
在構建標籤處理器是,須要在構建目錄中有Servlet API及JSP API(servlet-api.jar和jsp-api.jar)這兩個文件。自定義標籤由組件處理器(WEB-INF/classes)及標籤描述器(WEB-INF中的.tld)文件組成。一樣也須要taglib指令使用自定義標籤。
能夠把自定義的標籤處理器以及標籤描述器打包到jar包中,並指定絕對的URI,這樣就能夠把它像JSTL同樣發佈出來。
演進7:編寫自定義標籤是一件冗長瑣碎的事,須要編寫並變異一個標籤處理類還要在標籤庫中進行描述。經過tag file的方式,無須辨析標籤處理類和標籤庫描述文件也可以自定義標籤。tag file使用前不須要編譯,也不須要描述文件。 |
tag file無需提早編譯且只須要JSP語法就能夠。一個tag file擁有指令,腳本,EL,動做元素以及自定義標籤,一個tag file以tag和tagx爲後綴,它們能夠包含其餘資源,一個被其餘文件包含的tag file應該以tagf爲後綴。
tag文件必須放在路徑的WEB-INF/tags目錄下才能生效,和標籤處理器同樣,tag文件也能夠打包成jar文件。
解耦2:Servlet提供了一系列的事件和事件監聽藉口,上層的servlet/JSP應用可以經過調用這些API進行事件驅動開發。 |
監聽器都繼承自java.util.Event對象,監聽器接口能夠分爲ServletContext,HttpSession和ServletRequest。監聽器即一組動做的接口。編寫一個監聽器,只須要寫一個java類來實現對應的監聽器接口就能夠了,而後經過@WebListener註解或者部署描述文檔中增長listener元素進行註冊。
演進8:使用Filter來攔截Request的請求,在用戶的請求訪問資源前處理ServletRequest以及ServletResponse能夠實現日誌記錄,加解密,session檢查和圖像文件保護。 |
Filter實現須要實現javax.servlet.Filter接口,須要實現init(),doFilter(),destroy()方法。Filter的配置能夠經過@WebFilter或部署描述中的filter元素進行配置。Filter的使用須要考慮到Filter Chain的實現順序和規則,在部署描述符中,先配置的先執行。
演進9: 修飾Request和Response實現Decorator模式 |
演進10: Servlet或者Filter佔用請求處理縣城,若是任務須要很長時間才能完成,當用戶的併發請求超過縣城樹,容器會沒有可用的線程。Servlet使用超時時間處理異步請求,釋放正在等待完成的線程。 |
演進11: 儘管能夠經過註解進行配置,可是在須要更加精細配置的狀況下,部署描述符依然是須要的。部署描述符必須被命名爲web.xml而且位於WEB-INF目錄下,Java類必須放在WEB-INF/classes目錄下,而Java的類庫必須位於WEB-INF/lib目錄下。全部的應用資源必須打包成.war爲後綴的JAR文件。 |
演進12: web fragment能夠實如今已有的web應用中部署插件和框架。 |
3、 Spring MVC的實現
演進13: Servlet的動態加載能夠實如今不重啓web應用的前提下,添加新的web對象,Servlet容器加載器能夠以插件形式發佈應用而不須要修改部署描述,對框架的使用特別有用。 |
ServletContext接口中提供的(建立,註冊,使用)(Filter,Listener,Servlet)的方法。
initializer庫是一個插件化的框架,有兩個資源MyServletContainerInitializer類以及javax.servlet.ServletContainerInitializer的元文件,這個元文件必須放在WEB-INF/services目錄下,這個元文件只有一行:initializer.MyServletContainerInitializer的實現類名。
演進13: Spring做爲開源的輕量級企業級應用開發框架,提供了依賴注入方法的實現。依賴注入是一種代碼可測試性的解決方案。 |
簡單來講,有兩個組件A和B,A依賴於B,假定A是一個類且又一個方法使用到了B,那麼A必須先得到組件B的實例引用。Spring的依賴注入會先建立B的實例,再建立A的實例,而後把B注入到A的實例中。
Spring XML的配置寫在spring-config.xml文件中,配置文件能夠是一份,也能夠分解爲多份以支持模塊化的配置,既能夠經過主配置文件讀取多份配置文件,也能夠在其餘配置文件中讀取主配置文件。
Spring建立控制反轉容器能夠經過構造器的方式,也能夠是setter方法。
解耦3:Spring MVC模式實現了Web應用開發的模型2方式 |
一個MVC模式的應用包含模型,視圖和控制器三個部分。視圖負責應用的展現。模型封裝了應用的數據和業務邏輯,控制器負責接收用戶輸入,改變模型以及調整視圖的顯示。
Spring MVC使用Servlet充當控制器,Structs2使用Filter充當控制器。大部分都採用JSP頁面做爲視圖。模型採用POJO(Plain Old Java Object),在實踐中會採用一個JavaBean來持有模型的狀態,並將業務邏輯放到一個Action類中,一個JavaBean必須擁有一個無參的構造器,經過getter/setter訪問參數,同時支持持久化。
基本的Spring MVC項目結構包含:
1) 一個Product類,做爲product的領域對象,Product類須要實現java.io.Serializable接口,由於須要保存在HttpSession中。
2) 一個ProductForm類,封裝了HTML表單的輸入項,ProductForm類魚Product類類似,能夠起到不將ServletRequest這個Servlet層的對象暴露給其它層的目的,同時也能夠在數據校驗失敗時,表單對象將用於保存和展現用戶在原始表單上的輸入。
3) 一個ControolerServlet類做爲控制器。
4) 兩個JSP頁面做爲view。
解耦4:解耦控制器代碼,若是將業務邏輯代碼都寫在Servlet控制器中是哪一個,這個Servlet類將隨着應用複雜度的增長而不斷膨脹,應當將業務邏輯代碼提取到獨立的controller類中。 |
InputProductController類和SaveProductController類都實現了Controller接口,這使得Controller Servlet變得更加專一,就像一個Dispatcher,而非一個controller,即DispatcherServlet用來進行controller的分派。
演進14: 在Web應用執行action時,須要進行輸入的校驗,編程式的校驗經過編碼進行用戶輸入校驗,聲明式提供包含教研規則的XML文檔或者屬性文件。 |
演進15:在應用MVC時,能夠在Controller類中調用後端業務邏輯。一般後段封裝了複雜的邏輯service類,在service類中,能夠實例化一個DAO類來訪問數據庫。在Spring環境中,Service對象能夠自動被注入到Controller實例中,而DAO對象能夠自動被注入到Service對象中。 |
4、 Spring MVC的優點
採用Spring MVC的優點:
1) 不須要編寫DispatcherServlet;
2) 基於XML的文件配置不須要從新編譯;
3) 能夠實例化控制器,並根據用戶的輸入來構造bean;
4) 能夠自動綁定用戶輸入,並正確進行數據類型的轉換;
5) 能夠進行用戶輸入的校驗,能夠重定向回輸入表單,支持編程式校驗和聲明式校驗;
6) 做爲Spring框架的一部分,能夠實現其餘Spring提供的功能;
7) 支持國際化和本地化;
8) 支持多視圖技術(JSP,FreeMarker,Velocity)。
演進16:Spring MVC自帶一個開箱即用的Dispatcher Servlet。並提供了Controller接口並公開了handleRequest方法。 |
要使用這個Servlet,須要在部署描述符中進行配置,而且會尋找一個應用程序的WEB-INF目錄下的配置文件servletName-servlet.xml。controller須要實現org.springframework.web.servlet.mvc.Controller,Controller接口的實現類職能處理一個單一動做。
同時也須要添加Spring MVC所須要的JAR文件。
演進17:Spring MVC使用視圖解析器負責解析視圖,能夠經過在配置文件中定義一個ViewResolver來配置試圖解析器。 |
springmvc-config.xml實現了Dispatcher Servlet和ViewResolver的配置。同時也須要在部署描述符中進行配置。
演進18:使用基於註解的控制器配置方法可使得一個控制器類處理多個Action。 |
@Controller註解類型用於指示Spring類的實例是一個控制器;
@RequestMapping註解能夠爲控制器內部的每個動做開發相應的處理方法;
@Autowired和@Service註解能夠將依賴注入到控制器內;
@ModelAttribute註解能夠實現使用Spring MVC每次調用請求處理方法產生的Model類型實例。
演進19:數據綁定是將用戶的輸入綁定到領域模型的一種特性,也再也不須要form bean這樣的表單bean。表單的標籤庫會輔助這樣的工做。 |
使用表單標籤庫須要聲明taglib指令。
演進20:因爲Spring自身的數據綁定是雜亂無章的,須要經過Converter和Formatter來完成數據的綁定。 |
Converter是通用元件,能夠在應用程序的任意層中使用,而Formatter則是專門爲Web層設計的。
須要實現的Converter接口和Formatter接口。同時也須要在springmvc-config.xml中進行註冊。
演進21:Converter和Formatter只是做用於field級,Validator能夠做用於object級。在調用Controller期間,將會有一個或者多個Formatter視圖進行field值的變換,一旦格式化成功,Validator將會介入。 |
Spring MVC中有兩種用戶輸入驗證方式:Spring自帶的驗證框架和JSR 303的實現。前者須要實現Validator接口,而且須要調用reject方法來添加錯誤;後者則是經過註解給對象屬性添加約束。這樣的驗證器不須要顯式註冊,但若是想要從某個屬性文件中獲取錯誤信息能夠在springmvc-config.xml中進行添加。
演進22:Spring MVC提供了國際化和本地方的支持,須要講文本元文件隔離成屬性文件。 |
將每個語言區域的文本元素都單獨保存在一個獨立的屬性文件中,每一個文件都包含key/value對,而且每一個key都衛衣表示一個特定語言區域對象。並在springmvc-config.xml中進行配置。