SpringMVC超詳細講解視圖和視圖解析器
SpringMVC-視圖和視圖解析器
1.基本介紹
在 springMVC 中的目標方法最終返回都是一個視圖(有各種視圖).
返回的視圖都會由一個視圖解析器來處理 (視圖解析器有很多種)
2.自定義視圖
1.為什麼要自定義視圖
在默認情況下,我們都是返回默認的視圖, 然後這個返回的視圖交由 SpringMVC 的 InternalResourceViewResolver 視圖處理器來處理的
<bean id="internalResourceViewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver"> <property name="prefix" value="/WEB-INF/pages/"/> <property name="suffix" value=".jsp"/> </bean>
在實際開發中,我們有時需要自定義視圖,這樣可以滿足更多更復雜的需求
2.自定義視圖實例-代碼實現
1.配置 springDispatcherServlet-servlet.xml , 增加自定義視圖解析器
<!-- 1. 配置自定義視圖解析器BeanNameViewResolver 2. BeanNameViewResolver可以去解析我們自定義的視圖 3. 配置 屬性 order, 表示視圖解析器執行的順序, 值越小, 優先級越高 4. 屬性 order 的默認值是最低優先級 ,值為 Integer.MAX_VALUE int LOWEST_PRECEDENCE = 2147483647 --> <bean class="org.springframework.web.servlet.view.BeanNameViewResolver"> <property name="order" value="99"/> </bean>
2.創建自定義視圖類
@Component(value = "view") public class MyView extends AbstractView { @Override protected void renderMergedOutputModel(Map<String, Object> model, HttpServletRequest request, HttpServletResponse response) throws Exception { //完成視圖渲染 //並且可以確定我們要跳轉的頁面 [請求轉發] /WEB-INF/pages/my_view.jsp System.out.println("進入到自己的視圖.."); Object o = model.get("name"); //llp System.out.println(o); //1. 下面就是進行請求轉發到 /WEB-INF/pages/my_view.jsp //2. /WEB-INF/pages/my_view.jsp 會被springmvc解析 // /springmvc/WEB-INF/pages/my_view.jsp request.getRequestDispatcher("/WEB-INF/pages/my_view.jsp").forward(request, response); } }
3.創建goodsHandler測試
@Controller @RequestMapping("goods") public class GoodsHandler { @RequestMapping(value = "/buy") public String byGoods(Map<String,Object> map){ map.put("name","llp"); return "view"; } }
4.創建my_view.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %> <html> <head> <title>my_view頁面</title> </head> <h1>進入到my_view頁面</h1> <p>是從自定義視圖來的..</p> <body> </body> </html>
5.測試效果
3.自定義視圖工作流程小結
自定義視圖-小結
自定義視圖: 創建一個 View 的 bean, 該 bean 需要繼承自 AbstractView, 並實現 renderMergedOutputModel 方法.
並把自定義 View 加入到 IOC 容器中
自定義視圖的視圖處理器,使用 BeanNameViewResolver, 這個視圖處理器也需要配置 到 ioc 容器
BeanNameViewResolver 的調用優先級需要設置一下,設置 order 比 Integer.MAX_VAL 小的值. 以確保其在 InternalResourceViewResolver 之前被調用
自定義視圖-工作流程
SpringMVC 調用目標方法, 返回自定義 View 在 IOC 容器中的 id
SpringMVC 調用 BeanNameViewResolver 解析視圖: 從 IOC 容器中獲取 返回 id 值對應的 bean, 即自定義的 View 的對象
SpringMVC 調用自定義視圖的 renderMergedOutputModel 方法渲染視圖
說明: 如果在 SpringMVC 調用目標方法, 返回自定義 View 在 IOC 容器中的 id, 不存在, 則仍然按照默認的視圖處理器機制處理
如果將默認視圖解析器設置的優先級比自定義視圖高,默認視圖解析器不管頁面是否存在都會直接返回,不會在走自定義視圖解析器。
自定義 View在IOC容器中的 id存在時,自定義視圖解析器執行流程:
自定義 View在IOC容器中的 id不存在時,自定義視圖解析器執行流程
通用會先執行BeanNameViewResolver,可以看到此時,容器中不包含id=”view”的bean
回到DispatcherServlet我們可以看到 List viewResolvers列表包含兩個視圖解析器,一個時自定義的視圖還有一個是默認的視圖解析器。如果返回的自定義視圖為空,會繼續遍歷最終初始化默認視圖解析器從而按照默認視圖解析器的處理機制繼續執行,不管/WEB-INF/pages/view.jsp 是否存在程序都會返回
4.目標方法直接指定轉發或重定向
1.使用實例
目標方法中指定轉發或者重定向
默認返回的方式是請求轉發,然後用視圖處理器進行處理,比如在目標方法中這樣寫:
@Controller public class LoginServlet { @RequestMapping(value = "/login") public String login(){ System.out.println("login...."); return "login_ok"; } }
也可以在目標方法直接指定重定向或轉發的 url 地址
如果指定重定向,不能定向到 /WEB-INF 目錄中
應用實例-代碼實現
修改 GoodsHandler.java, 增加方法 order()
/** * 演示直接指定要請求轉發的或者是重定向的頁面 * @return */ @RequestMapping(value = "/order") public String order() { System.out.println("=======order()====="); //請求轉發到 /WEB-INF/pages/my_view.jsp //下面的 /WEB-INF/pages/my_view.jsp 被解析成 /springmvc/WEB-INF/pages/my_view.jsp //return "forward:/WEB-INF/pages/my_view.jsp"; //return "forward:/aaa/bbb/ok.jsp"; //直接指定要重定向的頁面 //1. 對於重定向來說,不能重定向到 /WEB-INF/ 目錄下 //2. redirect 關鍵字,表示進行重定向 //3. /login.jsp 在服務器解析 /springmvc/login.jsp return "redirect:/login.jsp"; // /WEB-INF/pages/my_view.jsp 被解析 /springmvc/WEB-INF/pages/my_view.jsp //return "redirect:/WEB-INF/pages/my_view.jsp"; }
2.修改view.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %> <html> <head> <title>自定義視圖測試</title> </head> <body> <h1>自定義視圖測試</h1> <a href="goods/buy" rel="external nofollow" >點擊到自定義視圖-</a><br/> <a href="goods/order" rel="external nofollow" >測試在目標方法中指定請求轉發或者重定向的頁面-</a><br/> </body> </html>
2.指定請求轉發流程-Debug源碼
1.SpirngMVC默認情況下
/** * 演示直接指定要請求轉發的或者是重定向的頁面 * @return */ @RequestMapping(value = "/order") public String order() { System.out.println("=======order()====="); return "my_view.jsp"; }
可以看到默認轉發是走的InternalResourceView默認視圖解析器,beanName=my_view.jsp 對應的實例則是InternalResourceView
最終執行到InternalResourceView的renderMergedOutputModel方法中dispatcherPath=/WEB-INF/pages/my_view.jsp.jsp進行請求轉發
2.指定forward關鍵字
/** * 演示直接指定要請求轉發的或者是重定向的頁面 * @return */ @RequestMapping(value = "/order") public String order() { System.out.println("=======order()====="); return "forward:/WEB-INF/pages/my_view.jsp"; }
在DispatcherServlet中我們可以看到,當我們指定瞭forward關鍵字時,viewResolver還是默認的InternalResourceView,和不指定forward關鍵字不同的地方子在於,指定瞭關鍵字後beanName變成瞭forward: 而url則是我們後面指定的路徑,因此我們在使用forward關鍵字進行請求轉發時,SpringMVC底層會根據程序員在後面指定的路徑進行請求轉發,如果設置的路徑不存在則會拋出404NOTFOUND異常
3.指定重定向流程-Debug源碼
/** * 演示直接指定要請求轉發的或者是重定向的頁面 * @return */ @RequestMapping(value = "/order") public String order() { System.out.println("=======order()====="); return "redirect:/login.jsp"; }
在前面我們配置瞭自定義視圖和默認視圖解析器,可以看到這裡走的是默認的視圖解析器。redirect: 被作為beanName, RedirectView則是對應的bean實例
從裡可以看出在SpringMvc中,重定向"redirect:/login.jsp" ,斜杠被瀏覽器解析成ip:port,而在服務端springmvc底層RedirectView會解析成“/springmvc/login.jsp”
RedirectView
@Override protected void renderMergedOutputModel(Map<String, Object> model, HttpServletRequest request, HttpServletResponse response) throws IOException { String targetUrl = createTargetUrl(model, request); targetUrl = updateTargetUrl(targetUrl, model, request, response); // Save flash attributes RequestContextUtils.saveOutputFlashMap(targetUrl, request, response); // Redirect 重定向 sendRedirect(request, response, targetUrl, this.http10Compatible); }
到此這篇關於SpringMVC超詳細講解視圖和視圖解析器的文章就介紹到這瞭,更多相關SpringMVC視圖內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 使用springmvc配置視圖解析器和重定向方式
- SpringMVC響應視圖和結果視圖詳解
- SpringMVC視圖轉發重定向區別及控制器詳解
- 淺談springMVC中controller的幾種返回類型
- SpringMVC數據頁響應ModelAndView實現頁面跳轉